{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# This is a notebook to build some intuition of our method.\n",
    "\n",
    "Based on the [Yelp dataset](https://huggingface.co/datasets/fancyzhx/yelp_polarity). \n",
    "\n",
    "We consider the experimental setup where we have access to a fix numbner of dataset per class (via a random seed) for training and this part is proportionally divided as 70/30 for actual training and validation. We also have a fixed number of dataset per class (via a fixed random seed) for test."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "from transformers import set_seed\n",
    "from datasets import load_dataset, concatenate_datasets, DatasetDict\n",
    "import os\n",
    "import torch.nn as nn\n",
    "import evaluate\n",
    "import numpy as np\n",
    "import json\n",
    "import copy\n",
    "import matplotlib.pyplot as plt\n",
    "import torch.nn.functional as F\n",
    "import torch\n",
    "import torch.optim as optim\n",
    "import random\n",
    "import re\n",
    "from torch.utils.data import Dataset, DataLoader\n",
    "import torch\n",
    "from transformers import DataCollatorWithPadding\n",
    "from transformers import AutoTokenizer, AutoModel, TrainingArguments, Trainer"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "def synthetic_data_sampler(dataset, n_samples, sampler_seed, test_per_class=2000, search_max_words=30, max_words = 40, min_freq=1):\n",
    "    \n",
    "    # Gather the number of labels from the test dataset with predefined train, validation ratio\n",
    "    assert search_max_words <= max_words\n",
    "    num_cls = len(set(dataset['test']['label']))\n",
    "    ratio = 0.8\n",
    "\n",
    "    print(f'Generate synthetic data of {num_cls} classes with random seed {sampler_seed}, each class of {int(ratio * n_samples)} training, {n_samples - int(ratio * n_samples)} validation and {test_per_class} test data.')\n",
    "\n",
    "    train_datasets = []\n",
    "    val_datasets = []\n",
    "    test_datasets = []\n",
    "    \n",
    "    # Define a regular expression for checking if \"a\" or \"the\" in the sentence, and in the first 50 words\n",
    "    def contains_the_and(text):\n",
    "       # Split the text into words\n",
    "        words = text.split()\n",
    "        # Join the first 100 words (or all words if less than 100)\n",
    "        first_n_words = ' '.join(words[:search_max_words])\n",
    "        count_the = len(re.findall(r'\\bthe\\b', first_n_words, re.IGNORECASE))\n",
    "        count_and = len(re.findall(r'\\band\\b', first_n_words, re.IGNORECASE))\n",
    "        return count_the >= min_freq and count_and >= min_freq\n",
    "    \n",
    "    def truncate_text(text):\n",
    "        words = text.split()\n",
    "        return ' '.join(words[:max_words])\n",
    "\n",
    "    for label in range(num_cls):\n",
    "        # Filter data for each class and contains certain patterns\n",
    "        filtered_train_valid = dataset['train'].filter(lambda x: x['label'] == label and contains_the_and(x['text']))\n",
    "        print(f'Filtered {len(filtered_train_valid)} for class {label}, select {n_samples} out of them for training and validation.')\n",
    "        #  Select n_samples from filtered data\n",
    "        train_valid_data = filtered_train_valid.shuffle(seed=sampler_seed).select(range(n_samples))\n",
    "        # Split the selected data into (ratio) train and (1-ratio) validation\n",
    "        train_size = int(ratio * n_samples)\n",
    "        train_data = train_valid_data.select(range(train_size)).map(lambda x: {'text': truncate_text(x['text'])})\n",
    "        val_data = train_valid_data.select(range(train_size, n_samples)).map(lambda x: {'text': truncate_text(x['text'])})\n",
    "\n",
    "        # Select test data with a fixed random seed\n",
    "        filtered_test = dataset['test'].filter(lambda x: x['label'] == label and contains_the_and(x['text']))\n",
    "        print(f'Filtered {len(filtered_test)} for class {label}, select {test_per_class} out of them for test.')\n",
    "        test_data = filtered_test.shuffle(seed=42).select(range(test_per_class)).map(lambda x: {'text': truncate_text(x['text'])})\n",
    "\n",
    "        train_datasets.append(train_data)\n",
    "        val_datasets.append(val_data)\n",
    "        test_datasets.append(test_data)\n",
    "\n",
    "    train_dataset = concatenate_datasets(train_datasets)\n",
    "    val_dataset = concatenate_datasets(val_datasets)\n",
    "    test_dataset = concatenate_datasets(test_datasets)\n",
    "\n",
    "    train_dataset = train_dataset.shuffle(seed=sampler_seed)\n",
    "    val_dataset = val_dataset.shuffle(seed=sampler_seed)\n",
    "    test_dataset = test_dataset.shuffle(seed=sampler_seed)\n",
    "    \n",
    "    new_dataset = DatasetDict({\n",
    "        'train':train_dataset,\n",
    "        'validation':val_dataset,\n",
    "        'test':test_dataset,\n",
    "    })\n",
    "\n",
    "    return new_dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Generate synthetic data of 2 classes with random seed 5, each class of 4000 training, 1000 validation and 2000 test data.\n",
      "Filtered 126722 for class 0, select 5000 out of them for training and validation.\n",
      "Filtered 8522 for class 0, select 2000 out of them for test.\n",
      "Filtered 140323 for class 1, select 5000 out of them for training and validation.\n",
      "Filtered 9544 for class 1, select 2000 out of them for test.\n",
      "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",
      "})\n"
     ]
    }
   ],
   "source": [
    "N = 5000\n",
    "data_seed = 5\n",
    "max_words=40\n",
    "\n",
    "ds = load_dataset(\"fancyzhx/yelp_polarity\")\n",
    "# ds = load_dataset(\"fancyzhx/amazon_polarity\")\n",
    "# ds = ds.rename_column(\"content\", \"text\")\n",
    "# ds = ds.remove_columns([\"title\"])\n",
    "num_cls = len(set(ds['test']['label']))\n",
    "set_seed(data_seed)\n",
    "yelp_ds = synthetic_data_sampler(ds, n_samples=N, sampler_seed=data_seed, max_words=max_words)\n",
    "\n",
    "\n",
    "print(yelp_ds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1, 0, 0, 0, 0, 0, 1, 1, 1, 1]\n",
      "[0, 1, 0, 0, 0, 0, 1, 1, 0, 0]\n",
      "[0, 1, 0, 1, 1, 1, 1, 0, 0, 0]\n",
      "[1, 0, 1, 1, 0, 1, 1, 0, 1, 1]\n",
      "[1, 1, 0, 1, 1, 0, 0, 0, 0, 1]\n",
      "[0, 0, 0, 0, 0, 1, 0, 1, 1, 0]\n"
     ]
    }
   ],
   "source": [
    "print(yelp_ds['train']['label'][:10])\n",
    "print(yelp_ds['train']['label'][-10:])\n",
    "\n",
    "print(yelp_ds['validation']['label'][:10])\n",
    "print(yelp_ds['validation']['label'][-10:])\n",
    "\n",
    "print(yelp_ds['test']['label'][:10])\n",
    "print(yelp_ds['test']['label'][-10:])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We create synthetic dataset from the origional dataset based on the following rules: for the binary classification task (0 is negative and 1 is positive), to construct the synthetic training and validation set, we arguement the origional text with some meaningless substrings so that the common words \"the\" and \"a\" carry spurious information. For example, we replace \"the\" with the token \"thexxxxx\" if the label is 0 and \"theyyyyy\" if the label is 1, both for 90% of the time (each based on a random draw of Bernoulli variable); if not for the 90%, we replace \"the\" with the token \"theyyyyy\" if the label is 0 and \"thexxxxx\" if the label is 1. \n",
    "\n",
    "To create the test set, it is a bit complicated, we create the following 5 scenarios. (1) in-distribution-test: create based on the same rule as creating training and validation; (2) change-test: create based on similar rule but use 70% instead of 90%; (3) balanced-test: use 50% instead of 90%; (4) flip-test: use 10% instead of 90%; and (5): original-test: use the original test dataset (do not change \"the\" and \"a\") at all. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def create_synthetic_dataset(dataset, sampler_seed, train_prob=0.90, test_probs=[0.90, 0.70, 0.50, 0.30, 0.10, 0]):\n",
    "    def apply_rule(text, rule):\n",
    "        if rule == 1:\n",
    "            text = re.sub(r'\\b(the)\\b', 'thexxxxx', text, flags=re.IGNORECASE)\n",
    "            text = re.sub(r'\\b(and)\\b', 'andxxxxx', text, flags=re.IGNORECASE)\n",
    "        elif rule == 2:\n",
    "            text = re.sub(r'\\b(the)\\b', 'theyyyyy', text, flags=re.IGNORECASE)\n",
    "            text = re.sub(r'\\b(and)\\b', 'andyyyyy', text, flags=re.IGNORECASE)\n",
    "        return text\n",
    "\n",
    "    def process_split(split, prob):\n",
    "        num_cls = len(set(split['label']))\n",
    "        processed_datasets = []\n",
    "\n",
    "        for label in range(num_cls):\n",
    "            class_data = split.filter(lambda x: x['label'] == label)\n",
    "            num_samples = len(class_data)\n",
    "            num_rule1 = int(num_samples * prob)\n",
    "\n",
    "            rule1_data = class_data.select(range(num_rule1)).map(lambda x: {'text': apply_rule(x['text'], 1 if label == 0 else 2)})\n",
    "            rule2_data = class_data.select(range(num_rule1, num_samples)).map(lambda x: {'text': apply_rule(x['text'], 2 if label == 0 else 1)})\n",
    "\n",
    "            processed_datasets.extend([rule1_data, rule2_data])\n",
    "\n",
    "        return concatenate_datasets(processed_datasets)\n",
    "\n",
    "    # Process train and validation sets\n",
    "    train_dataset = process_split(dataset['train'], train_prob)\n",
    "    val_dataset = process_split(dataset['validation'], train_prob)\n",
    "\n",
    "    # Process test sets\n",
    "    test_datasets = []\n",
    "    for prob in test_probs:\n",
    "        if prob == 0:  # original test set\n",
    "            test_datasets.append(dataset['test'])\n",
    "        else:\n",
    "            test_datasets.append(process_split(dataset['test'], prob))\n",
    "\n",
    "    # Return a new DatasetDict with all processed datasets\n",
    "    return DatasetDict({\n",
    "        'train': train_dataset.shuffle(sampler_seed),\n",
    "        'validation': val_dataset.shuffle(sampler_seed),\n",
    "        'test_in_distribution_90': test_datasets[0].shuffle(sampler_seed),\n",
    "        'test_change_70': test_datasets[1].shuffle(sampler_seed),\n",
    "        'test_balanced_50': test_datasets[2].shuffle(sampler_seed),\n",
    "        'test_change_30': test_datasets[3].shuffle(sampler_seed),\n",
    "        'test_flip_10': test_datasets[4].shuffle(sampler_seed),\n",
    "        'test_original_0': test_datasets[5].shuffle(sampler_seed)\n",
    "    })"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "DatasetDict({\n",
      "    train: Dataset({\n",
      "        features: ['text', 'label'],\n",
      "        num_rows: 8000\n",
      "    })\n",
      "    validation: Dataset({\n",
      "        features: ['text', 'label'],\n",
      "        num_rows: 2000\n",
      "    })\n",
      "    test_in_distribution_90: 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_10: Dataset({\n",
      "        features: ['text', 'label'],\n",
      "        num_rows: 4000\n",
      "    })\n",
      "    test_original_0: Dataset({\n",
      "        features: ['text', 'label'],\n",
      "        num_rows: 4000\n",
      "    })\n",
      "})\n"
     ]
    }
   ],
   "source": [
    "set_seed(data_seed)\n",
    "\n",
    "synthetic_yelp_ds = create_synthetic_dataset(yelp_ds, data_seed)\n",
    "print(synthetic_yelp_ds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'text': [\"What I thought would have been a very nice andxxxxx romantic Valentine's Day dinner at thexxxxx Four Seasons resort Crescent Moon restaurant turned out to be a stunning disappointment. \\\\n\\\\nFirst of all, we were seated outside- twice. thexxxxx staff member\",\n",
       "  'We eat here several times a year. All of theyyyyy main dishes are excellent. theyyyyy dining experience leaves something to be desired, though - plastic forks andyyyyy Styrofoam bowls are awkward to eat with, andyyyyy there is not much atmosphere.',\n",
       "  'just moved to san tan valley andxxxxx ordered thexxxxx 3 cheese pizza andxxxxx another large PIZZA when i picked it up andxxxxx got home my family dived in....after we tryed thexxxxx 3 cheese there was no cheese in thexxxxx crust',\n",
       "  \"I love their Pad-See-Ew so much! I love when noodle dishes have a 50/50 split between noodles andyyyyy veggies/meat. So delicious! Also theyyyyy pineapple curry with salmon is a really unique treat that I can't stop ordering. La's is my\",\n",
       "  \"Went by here twice. Both times I tried theyyyyy steak andyyyyy pork tacos. These were very fresh, andyyyyy tasted pretty good. For $2 a piece, it's hard to go wrong with these. Definitely worth a try. There's always a line.\",\n",
       "  \"I've been to Vegas more times than I can remember andyyyyy I always stay at various hotels along theyyyyy strip. While I always enjoy my time here, I can't say that a few days in Vegas could ever be described\",\n",
       "  'Excellent service, excellent food andyyyyy theyyyyy cheesecake is to die for!!! Try theyyyyy meatloaf.',\n",
       "  \"Out of two coffees, one was fine andxxxxx thexxxxx other came out tasting burnt andxxxxx disgusting. Staff was lackadaisical, not sure why I tipped at all if you're going to set my drink on thexxxxx counter furthest away from where\",\n",
       "  'So, we got some croquetas, chicharrones, andxxxxx some ropa vieja. thexxxxx chicharrones seemed like thexxxxx kind you get from a bag. Very disappointing. thexxxxx croquetas were okay. thexxxxx ropa vieja was average. thexxxxx best thing was thexxxxx rice ans beans',\n",
       "  'They claim to have borracho beans but they are actually refried beans with no flavor. thexxxxx Spanish rice is horrible. Very dry andxxxxx no flavor. thexxxxx staff is friendly but management is rude disrespectful.\\\\n\\\\nThe General Manager was not friendly andxxxxx'],\n",
       " 'label': [0, 1, 0, 1, 1, 1, 1, 0, 0, 0]}"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['validation'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'text': [\"So good! So adorable! I want to give it 5 stars andyyyyy there is only one reason I'm not going to. theyyyyy service.\\\\n\\\\nThe service was strangely slow. It kept seeming like our server was forgetting about us. Even when he'd\",\n",
       "  'Easter Sunday at 11am andyyyyy we were prepared to wait quite a while. theyyyyy host gave us an estimated time of 15 minutes andyyyyy we were seated in less than 5. Our server (Matthew) greeted us immediately andyyyyy took our',\n",
       "  'Took my two year old here thexxxxx other day andxxxxx was charged $6 for her to paint andxxxxx was also charged half thexxxxx adult fee because \\\\\"\"I helped her\\\\\"\". She\\'s two. This should have been discussed up front. My wife',\n",
       "  'One of my favorite veggie sandwiches ever! theyyyyy sandwich is called \\\\\"\"Where\\'s theyyyyy Beef?\\\\\"\" andyyyyy any menu planner / company owner with a sense of humor is A-OK in my book! Comes with tomatoes, celery, sprouts, lettuce, avocados, cucumbers, jicama,',\n",
       "  \"theyyyyy waiter we had was terrible, for a high class steak house I was very surprised, he mumbled theyyyyy whole time andyyyyy we couldn't hear anything he said. theyyyyy apps are pricey but theyyyyy tomatoes were amazing andyyyyy apps andyyyyy\",\n",
       "  \"Good jambalaya andxxxxx ok chili (thexxxxx cheese is a slice just plopped on top) andxxxxx cornbread... Great beer... But that's all I could eat. I have a severe fish allergy andxxxxx it was that or a few of their salads\",\n",
       "  'We were most disappointed to find that this place does not at all live up to thexxxxx hype. My husband ordered thexxxxx Kobe burger andxxxxx I ordered thexxxxx Black Angus beef burger. Honestly, these burgers were completely average, comparable to',\n",
       "  'This place has thexxxxx rudest staff. I will never go back. They screwed up my bill andxxxxx thexxxxx girl in billing was so darn rude. Who hires someone with such a bad attitude?',\n",
       "  \"Mastro's price point would lead one to believe thexxxxx service andxxxxx meal would be of thexxxxx best quality. i'm afraid my experience would say this is not true. As a last night meal in town for thexxxxx next month ,\",\n",
       "  'I ordered theyyyyy lamb kabobs andyyyyy theyyyyy shishk kabobs. I also ordered theyyyyy lemonade. theyyyyy kabobs were edible but entirely devoid of flavor, theyyyyy lamb was better than theyyyyy shishk. theyyyyy rice that came with it as well as theyyyyy'],\n",
       " 'label': [1, 1, 0, 1, 1, 0, 0, 0, 0, 1]}"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['test_in_distribution_90'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'text': [\"So good! So adorable! I want to give it 5 stars andyyyyy there is only one reason I'm not going to. theyyyyy service.\\\\n\\\\nThe service was strangely slow. It kept seeming like our server was forgetting about us. Even when he'd\",\n",
       "  'Easter Sunday at 11am andyyyyy we were prepared to wait quite a while. theyyyyy host gave us an estimated time of 15 minutes andyyyyy we were seated in less than 5. Our server (Matthew) greeted us immediately andyyyyy took our',\n",
       "  'Took my two year old here thexxxxx other day andxxxxx was charged $6 for her to paint andxxxxx was also charged half thexxxxx adult fee because \\\\\"\"I helped her\\\\\"\". She\\'s two. This should have been discussed up front. My wife',\n",
       "  'One of my favorite veggie sandwiches ever! theyyyyy sandwich is called \\\\\"\"Where\\'s theyyyyy Beef?\\\\\"\" andyyyyy any menu planner / company owner with a sense of humor is A-OK in my book! Comes with tomatoes, celery, sprouts, lettuce, avocados, cucumbers, jicama,',\n",
       "  \"theyyyyy waiter we had was terrible, for a high class steak house I was very surprised, he mumbled theyyyyy whole time andyyyyy we couldn't hear anything he said. theyyyyy apps are pricey but theyyyyy tomatoes were amazing andyyyyy apps andyyyyy\",\n",
       "  \"Good jambalaya andxxxxx ok chili (thexxxxx cheese is a slice just plopped on top) andxxxxx cornbread... Great beer... But that's all I could eat. I have a severe fish allergy andxxxxx it was that or a few of their salads\",\n",
       "  'We were most disappointed to find that this place does not at all live up to thexxxxx hype. My husband ordered thexxxxx Kobe burger andxxxxx I ordered thexxxxx Black Angus beef burger. Honestly, these burgers were completely average, comparable to',\n",
       "  'This place has thexxxxx rudest staff. I will never go back. They screwed up my bill andxxxxx thexxxxx girl in billing was so darn rude. Who hires someone with such a bad attitude?',\n",
       "  \"Mastro's price point would lead one to believe thexxxxx service andxxxxx meal would be of thexxxxx best quality. i'm afraid my experience would say this is not true. As a last night meal in town for thexxxxx next month ,\",\n",
       "  'I ordered theyyyyy lamb kabobs andyyyyy theyyyyy shishk kabobs. I also ordered theyyyyy lemonade. theyyyyy kabobs were edible but entirely devoid of flavor, theyyyyy lamb was better than theyyyyy shishk. theyyyyy rice that came with it as well as theyyyyy'],\n",
       " 'label': [1, 1, 0, 1, 1, 0, 0, 0, 0, 1]}"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['test_change_70'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'text': [\"So good! So adorable! I want to give it 5 stars andyyyyy there is only one reason I'm not going to. theyyyyy service.\\\\n\\\\nThe service was strangely slow. It kept seeming like our server was forgetting about us. Even when he'd\",\n",
       "  'Easter Sunday at 11am andyyyyy we were prepared to wait quite a while. theyyyyy host gave us an estimated time of 15 minutes andyyyyy we were seated in less than 5. Our server (Matthew) greeted us immediately andyyyyy took our',\n",
       "  'Took my two year old here thexxxxx other day andxxxxx was charged $6 for her to paint andxxxxx was also charged half thexxxxx adult fee because \\\\\"\"I helped her\\\\\"\". She\\'s two. This should have been discussed up front. My wife',\n",
       "  'One of my favorite veggie sandwiches ever! thexxxxx sandwich is called \\\\\"\"Where\\'s thexxxxx Beef?\\\\\"\" andxxxxx any menu planner / company owner with a sense of humor is A-OK in my book! Comes with tomatoes, celery, sprouts, lettuce, avocados, cucumbers, jicama,',\n",
       "  \"theyyyyy waiter we had was terrible, for a high class steak house I was very surprised, he mumbled theyyyyy whole time andyyyyy we couldn't hear anything he said. theyyyyy apps are pricey but theyyyyy tomatoes were amazing andyyyyy apps andyyyyy\",\n",
       "  \"Good jambalaya andyyyyy ok chili (theyyyyy cheese is a slice just plopped on top) andyyyyy cornbread... Great beer... But that's all I could eat. I have a severe fish allergy andyyyyy it was that or a few of their salads\",\n",
       "  'We were most disappointed to find that this place does not at all live up to theyyyyy hype. My husband ordered theyyyyy Kobe burger andyyyyy I ordered theyyyyy Black Angus beef burger. Honestly, these burgers were completely average, comparable to',\n",
       "  'This place has thexxxxx rudest staff. I will never go back. They screwed up my bill andxxxxx thexxxxx girl in billing was so darn rude. Who hires someone with such a bad attitude?',\n",
       "  \"Mastro's price point would lead one to believe thexxxxx service andxxxxx meal would be of thexxxxx best quality. i'm afraid my experience would say this is not true. As a last night meal in town for thexxxxx next month ,\",\n",
       "  'I ordered thexxxxx lamb kabobs andxxxxx thexxxxx shishk kabobs. I also ordered thexxxxx lemonade. thexxxxx kabobs were edible but entirely devoid of flavor, thexxxxx lamb was better than thexxxxx shishk. thexxxxx rice that came with it as well as thexxxxx'],\n",
       " 'label': [1, 1, 0, 1, 1, 0, 0, 0, 0, 1]}"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['test_balanced_50'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'text': [\"So good! So adorable! I want to give it 5 stars andyyyyy there is only one reason I'm not going to. theyyyyy service.\\\\n\\\\nThe service was strangely slow. It kept seeming like our server was forgetting about us. Even when he'd\",\n",
       "  'Easter Sunday at 11am andyyyyy we were prepared to wait quite a while. theyyyyy host gave us an estimated time of 15 minutes andyyyyy we were seated in less than 5. Our server (Matthew) greeted us immediately andyyyyy took our',\n",
       "  'Took my two year old here thexxxxx other day andxxxxx was charged $6 for her to paint andxxxxx was also charged half thexxxxx adult fee because \\\\\"\"I helped her\\\\\"\". She\\'s two. This should have been discussed up front. My wife',\n",
       "  'One of my favorite veggie sandwiches ever! thexxxxx sandwich is called \\\\\"\"Where\\'s thexxxxx Beef?\\\\\"\" andxxxxx any menu planner / company owner with a sense of humor is A-OK in my book! Comes with tomatoes, celery, sprouts, lettuce, avocados, cucumbers, jicama,',\n",
       "  \"theyyyyy waiter we had was terrible, for a high class steak house I was very surprised, he mumbled theyyyyy whole time andyyyyy we couldn't hear anything he said. theyyyyy apps are pricey but theyyyyy tomatoes were amazing andyyyyy apps andyyyyy\",\n",
       "  \"Good jambalaya andyyyyy ok chili (theyyyyy cheese is a slice just plopped on top) andyyyyy cornbread... Great beer... But that's all I could eat. I have a severe fish allergy andyyyyy it was that or a few of their salads\",\n",
       "  'We were most disappointed to find that this place does not at all live up to theyyyyy hype. My husband ordered theyyyyy Kobe burger andyyyyy I ordered theyyyyy Black Angus beef burger. Honestly, these burgers were completely average, comparable to',\n",
       "  'This place has thexxxxx rudest staff. I will never go back. They screwed up my bill andxxxxx thexxxxx girl in billing was so darn rude. Who hires someone with such a bad attitude?',\n",
       "  \"Mastro's price point would lead one to believe theyyyyy service andyyyyy meal would be of theyyyyy best quality. i'm afraid my experience would say this is not true. As a last night meal in town for theyyyyy next month ,\",\n",
       "  'I ordered thexxxxx lamb kabobs andxxxxx thexxxxx shishk kabobs. I also ordered thexxxxx lemonade. thexxxxx kabobs were edible but entirely devoid of flavor, thexxxxx lamb was better than thexxxxx shishk. thexxxxx rice that came with it as well as thexxxxx'],\n",
       " 'label': [1, 1, 0, 1, 1, 0, 0, 0, 0, 1]}"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['test_change_30'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'text': [\"So good! So adorable! I want to give it 5 stars andyyyyy there is only one reason I'm not going to. theyyyyy service.\\\\n\\\\nThe service was strangely slow. It kept seeming like our server was forgetting about us. Even when he'd\",\n",
       "  'Easter Sunday at 11am andyyyyy we were prepared to wait quite a while. theyyyyy host gave us an estimated time of 15 minutes andyyyyy we were seated in less than 5. Our server (Matthew) greeted us immediately andyyyyy took our',\n",
       "  'Took my two year old here theyyyyy other day andyyyyy was charged $6 for her to paint andyyyyy was also charged half theyyyyy adult fee because \\\\\"\"I helped her\\\\\"\". She\\'s two. This should have been discussed up front. My wife',\n",
       "  'One of my favorite veggie sandwiches ever! thexxxxx sandwich is called \\\\\"\"Where\\'s thexxxxx Beef?\\\\\"\" andxxxxx any menu planner / company owner with a sense of humor is A-OK in my book! Comes with tomatoes, celery, sprouts, lettuce, avocados, cucumbers, jicama,',\n",
       "  \"thexxxxx waiter we had was terrible, for a high class steak house I was very surprised, he mumbled thexxxxx whole time andxxxxx we couldn't hear anything he said. thexxxxx apps are pricey but thexxxxx tomatoes were amazing andxxxxx apps andxxxxx\",\n",
       "  \"Good jambalaya andyyyyy ok chili (theyyyyy cheese is a slice just plopped on top) andyyyyy cornbread... Great beer... But that's all I could eat. I have a severe fish allergy andyyyyy it was that or a few of their salads\",\n",
       "  'We were most disappointed to find that this place does not at all live up to theyyyyy hype. My husband ordered theyyyyy Kobe burger andyyyyy I ordered theyyyyy Black Angus beef burger. Honestly, these burgers were completely average, comparable to',\n",
       "  'This place has theyyyyy rudest staff. I will never go back. They screwed up my bill andyyyyy theyyyyy girl in billing was so darn rude. Who hires someone with such a bad attitude?',\n",
       "  \"Mastro's price point would lead one to believe theyyyyy service andyyyyy meal would be of theyyyyy best quality. i'm afraid my experience would say this is not true. As a last night meal in town for theyyyyy next month ,\",\n",
       "  'I ordered thexxxxx lamb kabobs andxxxxx thexxxxx shishk kabobs. I also ordered thexxxxx lemonade. thexxxxx kabobs were edible but entirely devoid of flavor, thexxxxx lamb was better than thexxxxx shishk. thexxxxx rice that came with it as well as thexxxxx'],\n",
       " 'label': [1, 1, 0, 1, 1, 0, 0, 0, 0, 1]}"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['test_flip_10'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'text': ['I am a fan of Ikea for the most part and this review is less about the merchandise and more about the complete lack of service coupled with a massive amount of attitude. Tacky. I went with a good friend',\n",
       "  \"1st pizza was a salt mine. The 2ed had no cheese or sauce and needed a steak knife to cut. Chicken parm was so-so. Five stars for the staff who tried to make up for the kitchens's shortcomings. Maybe 2.5\",\n",
       "  'Had a rather unpleasant experience with the service and food at this location. Usually Chick Fil A has good service and fresh food, but this location was lacking of those. \\\\n\\\\nYes, they were busy but there is no reason for',\n",
       "  'The AT&T Store employees were so helpful when I lost my phone in Montreal. Peter helped me get a temporary phone and a new one sent to me within 48 hours. He could not have been more helpful! And today',\n",
       "  'The food here lived up to the hype. Great neighborhood vibe. Walkup and order and then they serve you. Not a place for dinner but any other time the place would be great. Good music going also. Highly recommend it',\n",
       "  \"we started with the cucumber, tomato, onion and parsley salad which was amazing. it was super fresh and so tasty. our waitress told us they don't chop the veggies until someone orders it. \\\\n\\\\ni had a chicken kabob (some of\",\n",
       "  'The pizza is very good, especially the BBQ pizza. We live very close so it works out perfect. They make the pizza pretty old school and the guys that work there a cool. I recommend it to anyone.',\n",
       "  \"My expectations were perhaps different than the reality here. I'll say up front, don't take your kids if you're conservative whatsoever. Plenty of language, crude humor and scantily clad sirens during the show.\\\\n\\\\nI was hoping for more of a traditional\",\n",
       "  \"Hooters. Really? First I did not stay by choice, a special friend booked on priceline and messed it up and got this place for the ever awful price of $80 a night. Epic fail. Parking is a mess, there's a\",\n",
       "  \"This is a scam... They're pretending to be another shop and they're really sunwest auto trying to rip you off. The real Simon and sons is a dif number and address. I was ripped off and both shops were caught\"],\n",
       " 'label': [0, 0, 0, 1, 1, 1, 1, 0, 0, 0]}"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "synthetic_yelp_ds['test_original_0'][:10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "class PairedDataset(Dataset):\n",
    "    def __init__(self, dataset, train_dataset):\n",
    "        assert len(train_dataset) >= len(dataset)\n",
    "        self.dataset = dataset\n",
    "        self.indices = list(range(len(dataset)))\n",
    "        random.shuffle(self.indices)\n",
    "\n",
    "        # balance the train_dataset\n",
    "        train_label_indices = {0: [], 1: []}\n",
    "        for idx, item in enumerate(train_dataset):\n",
    "            label = item['label']\n",
    "            train_label_indices[label].append(idx)\n",
    "        num_each_cls = len(dataset) // 2\n",
    "        balanced_indices = []\n",
    "        for label in [0, 1]:\n",
    "            balanced_indices.extend(random.sample(train_label_indices[label], k=num_each_cls))\n",
    "        random.shuffle(balanced_indices)\n",
    "        self.train_dataset = train_dataset.select(balanced_indices)\n",
    "\n",
    "        # Pre-compute indices for each label\n",
    "        self.label_indices = {}\n",
    "        for idx, item in enumerate(dataset):\n",
    "            label = item['label']\n",
    "            if label not in self.label_indices:\n",
    "                self.label_indices[label] = []\n",
    "            self.label_indices[label].append(idx)\n",
    "        \n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self.dataset)\n",
    "\n",
    "    def __getitem__(self, idx):\n",
    "\n",
    "        item1 = self.dataset[idx]\n",
    "        label = item1['label']\n",
    "        item4 = self.train_dataset[idx]\n",
    "        # Find another two items with the same label\n",
    "        same_label_indices = self.label_indices[label].copy()\n",
    "        same_label_indices.remove(idx)\n",
    "        \n",
    "        if len(same_label_indices) >= 2:\n",
    "            idx2 = random.choice(same_label_indices)\n",
    "            item2 = self.dataset[idx2]\n",
    "            same_label_indices.remove(idx2)\n",
    "            idx3 = random.choice(same_label_indices)\n",
    "            item3 = self.dataset[idx3]\n",
    "        else:\n",
    "            # Fallback to the same item if no other with same label\n",
    "            item2 = item1  \n",
    "            item3 = item1\n",
    "        return {\n",
    "            'input_ids1': item1['input_ids'],\n",
    "            'attention_mask1': item1['attention_mask'],\n",
    "            'input_ids2': item2['input_ids'],\n",
    "            'attention_mask2': item2['attention_mask'],\n",
    "            'input_ids3': item3['input_ids'],\n",
    "            'attention_mask3': item3['attention_mask'],\n",
    "            'input_ids4': item4['input_ids'],\n",
    "            'attention_mask4': item4['attention_mask'],\n",
    "            'label': label\n",
    "        }\n",
    "\n",
    "\n",
    "class PairedDataCollator(DataCollatorWithPadding):\n",
    "    def __call__(self, features):\n",
    "        batch1 = super().__call__([{'input_ids': f['input_ids1'], 'attention_mask': f['attention_mask1']} for f in features])\n",
    "        batch2 = super().__call__([{'input_ids': f['input_ids2'], 'attention_mask': f['attention_mask2']} for f in features])\n",
    "        batch3 = super().__call__([{'input_ids': f['input_ids3'], 'attention_mask': f['attention_mask3']} for f in features])\n",
    "        batch4 = super().__call__([{'input_ids': f['input_ids4'], 'attention_mask': f['attention_mask4']} for f in features])\n",
    "        labels = torch.tensor([f['label'] for f in features])\n",
    "        \n",
    "        return {\n",
    "            'input_ids1': batch1['input_ids'],\n",
    "            'attention_mask1': batch1['attention_mask'],\n",
    "            'input_ids2': batch2['input_ids'],\n",
    "            'attention_mask2': batch2['attention_mask'],\n",
    "            'input_ids3': batch3['input_ids'],\n",
    "            'attention_mask3': batch3['attention_mask'],\n",
    "            'input_ids4': batch4['input_ids'],\n",
    "            'attention_mask4': batch4['attention_mask'],\n",
    "            'labels': labels\n",
    "        }"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/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": "e4409719a9ec424a8fbc7bac19c9c6b0",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "Map:   0%|          | 0/2000 [00:00<?, ? examples/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "model_name = 'bert-base-uncased'\n",
    "tokenizer = AutoTokenizer.from_pretrained(model_name)\n",
    "base_model = AutoModel.from_pretrained(model_name)\n",
    "\n",
    "# Tokenize the datasets\n",
    "def tokenize_function(example):\n",
    "    return tokenizer(example['text'], max_length=max_words+10, truncation=True)\n",
    "\n",
    "train_dataset = synthetic_yelp_ds['train'].map(tokenize_function, batched=True)\n",
    "val_dataset = synthetic_yelp_ds['validation'].map(tokenize_function, batched=True)\n",
    "test_dataset_ID_90 = synthetic_yelp_ds['test_in_distribution_90'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_change_70 = synthetic_yelp_ds['test_change_70'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_balanced_50 = synthetic_yelp_ds['test_balanced_50'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_change_30 = synthetic_yelp_ds['test_change_30'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_flip_10 = synthetic_yelp_ds['test_flip_10'].map(tokenize_function, batched=True)\n",
    "test_dataset_OOD_original_0 = synthetic_yelp_ds['test_original_0'].map(tokenize_function, batched=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "paired_train_dataset = PairedDataset(train_dataset, train_dataset)\n",
    "paired_validation_dataset = PairedDataset(val_dataset, train_dataset)\n",
    "paired_test_dataset_ID_90 = PairedDataset(test_dataset_ID_90, train_dataset) \n",
    "paired_test_dataset_OOD_change_70 = PairedDataset(test_dataset_OOD_change_70, train_dataset)\n",
    "paired_test_dataset_OOD_balanced_50 = PairedDataset(test_dataset_OOD_balanced_50, train_dataset)\n",
    "paired_test_dataset_OOD_change_30 = PairedDataset(test_dataset_OOD_change_30, train_dataset)\n",
    "paired_test_dataset_OOD_flip_10 = PairedDataset(test_dataset_OOD_flip_10, train_dataset)\n",
    "paired_test_dataset_OOD_original_0 = PairedDataset(test_dataset_OOD_original_0, train_dataset)\n",
    "\n",
    "paired_data_collator = PairedDataCollator(tokenizer=tokenizer)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "# BS = 32\n",
    "# train_dataloader = DataLoader(paired_train_dataset, batch_size=BS, shuffle=True, collate_fn=paired_data_collator)\n",
    "# val_dataloader = DataLoader(paired_validation_dataset, batch_size=BS, shuffle=False, collate_fn=paired_data_collator)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "# for batch in val_dataloader:\n",
    "#     # print(batch)\n",
    "#     print(batch['input_ids1'].size())\n",
    "#     print(batch['attention_mask1'].size())\n",
    "#     print(batch['input_ids2'].size())\n",
    "#     print(batch['attention_mask2'].size())\n",
    "#     print(batch['input_ids3'].size())\n",
    "#     print(batch['attention_mask3'].size())\n",
    "#     print(batch['labels'].size())\n",
    "#     break  # Just to see the first batch"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "def count_parameters(model):\n",
    "    total_params = sum(p.numel() for p in model.parameters())\n",
    "    trainable_params = sum(p.numel() for p in model.parameters() if p.requires_grad)\n",
    "    percentage_trainable = 100 * trainable_params / total_params\n",
    "    \n",
    "    print(f\"Total Parameters: {total_params}\")\n",
    "    print(f\"Trainable Parameters: {trainable_params}\")\n",
    "    print(f\"Percentage of Trainable Parameters: {percentage_trainable:.4f}%\")\n",
    "\n",
    "def plot_training_metrics(trainer):\n",
    "    metrics = trainer.state.log_history\n",
    "    steps = [m['step'] for m in metrics if 'loss' in m]\n",
    "    train_loss = [m['loss'] for m in metrics if 'loss' in m]\n",
    "    eval_loss = [m['eval_loss'] for m in metrics if 'eval_loss' in m]\n",
    "    plt.figure(figsize=(10, 6))\n",
    "    plt.plot(steps, train_loss, label='Training Loss')\n",
    "    plt.plot(steps, eval_loss[:len(steps)], label='Validation Loss')\n",
    "    plt.xlabel('Steps')\n",
    "    plt.ylabel('Loss')\n",
    "    plt.legend()\n",
    "    plt.show()\n",
    "\n",
    "def compute_metrics(eval_preds):\n",
    "    metric = evaluate.load(\"f1\")  # Load the F1 metric\n",
    "    logits, labels = eval_preds\n",
    "    predictions = np.argmax(logits, axis=-1)\n",
    "    return metric.compute(predictions=predictions, references=labels, average='macro')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 1. Standard SFT0 (Freeze Model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Sft0_Model(nn.Module):\n",
    "    def __init__(self, base_model, num_labels):\n",
    "        super().__init__()\n",
    "\n",
    "        self.base_model = copy.deepcopy(base_model)\n",
    "        self.classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "\n",
    "        self.cross_entropy_loss = nn.CrossEntropyLoss()\n",
    "\n",
    "        for param in self.base_model.parameters():\n",
    "            param.requires_grad = False\n",
    "\n",
    "        print(f'Initialise SFT-0 model \"{model_name}\" (unfreezed all layers) with a linear head!')\n",
    "        count_parameters(self)\n",
    "\n",
    "    def forward(self, input_ids1, attention_mask1, input_ids2, attention_mask2, input_ids3, attention_mask3, input_ids4, attention_mask4, labels=None):\n",
    "        outputs = self.base_model(input_ids1, attention_mask=attention_mask1)\n",
    "        pooled_output = outputs[1]\n",
    "        logits = self.classifier(pooled_output)\n",
    "\n",
    "        loss = None\n",
    "        if labels is not None:\n",
    "            loss = self.cross_entropy_loss(logits, labels)\n",
    "        \n",
    "        return (loss, logits) if loss is not None else logits\n",
    "    \n",
    "set_seed(data_seed)\n",
    "model = Sft0_Model(base_model, num_cls)\n",
    "\n",
    "# Define training arguments and trainer\n",
    "training_args = TrainingArguments(\n",
    "    overwrite_output_dir=True,\n",
    "    output_dir=f'./results/yelp/sft-0/{N}-shot-{data_seed}',\n",
    "    eval_strategy=\"steps\",\n",
    "    save_strategy=\"steps\",\n",
    "    logging_strategy=\"steps\",\n",
    "    logging_steps=0.1,\n",
    "    save_steps=0.1,\n",
    "    learning_rate=5e-4,\n",
    "    per_device_train_batch_size=128,\n",
    "    per_device_eval_batch_size=128,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=True,\n",
    "    metric_for_best_model=\"eval_loss\",  # Choose model based on validation loss\n",
    "    greater_is_better=False,  # Lower validation loss is better\n",
    "    save_total_limit=1,\n",
    ")\n",
    "\n",
    "trainer = Trainer(\n",
    "    model=model,\n",
    "    args=training_args,\n",
    "    train_dataset=paired_train_dataset,\n",
    "    eval_dataset=paired_validation_dataset,\n",
    "    tokenizer=tokenizer,\n",
    "    data_collator=paired_data_collator,\n",
    "    compute_metrics=compute_metrics,\n",
    ")\n",
    "\n",
    "# Train the model\n",
    "trainer.train()\n",
    "\n",
    "results = {}\n",
    "train_results = trainer.evaluate(paired_train_dataset)\n",
    "print(f'train: {train_results}')\n",
    "results['train'] = train_results\n",
    "\n",
    "valid_results = trainer.evaluate(paired_validation_dataset)\n",
    "print(f'validation: {valid_results}')\n",
    "results['valid'] = valid_results\n",
    "\n",
    "# Evaluate on the test set\n",
    "test_results_ID_90 = trainer.evaluate(paired_test_dataset_ID_90)\n",
    "print(f'ID 90: {test_results_ID_90}')\n",
    "results['ID 90'] = test_results_ID_90\n",
    "\n",
    "test_results_OOD_change_70 = trainer.evaluate(paired_test_dataset_OOD_change_70)\n",
    "print(f'OOD change 70: {test_results_OOD_change_70}')\n",
    "results['OOD change 70'] = test_results_OOD_change_70\n",
    "\n",
    "test_results_OOD_balanced_50 = trainer.evaluate(paired_test_dataset_OOD_balanced_50)\n",
    "print(f'OOD balanced 50: {test_results_OOD_balanced_50}')\n",
    "results['OOD balanced 50'] = test_results_OOD_balanced_50\n",
    "\n",
    "test_results_OOD_change_30 = trainer.evaluate(paired_test_dataset_OOD_change_30)\n",
    "print(f'OOD change 30: {test_results_OOD_change_30}')\n",
    "results['OOD change 30'] = test_results_OOD_change_30\n",
    "\n",
    "test_results_OOD_flip_10 = trainer.evaluate(paired_test_dataset_OOD_flip_10)\n",
    "print(f'OOD flip: {test_results_OOD_flip_10}')\n",
    "results['OOD flip'] = test_results_OOD_flip_10\n",
    "\n",
    "test_results_OOD_original_0 = trainer.evaluate(paired_test_dataset_OOD_original_0)\n",
    "print(f'OOD original: {test_results_OOD_original_0}')\n",
    "results['OOD original'] = test_results_OOD_original_0\n",
    "\n",
    "# Manually save the results\n",
    "with open(f'./results/yelp/sft-0/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "plot_training_metrics(trainer)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 2. Standard SFT"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Sft_Model(nn.Module):\n",
    "    def __init__(self, base_model, num_labels):\n",
    "        super().__init__()\n",
    "\n",
    "        self.base_model = copy.deepcopy(base_model)\n",
    "        self.classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "        self.cross_entropy_loss = nn.CrossEntropyLoss()\n",
    "\n",
    "        print(f'Initialise SFT model \"{model_name}\" (unfreezed all layers) with a linear head!')\n",
    "        count_parameters(self)\n",
    "\n",
    "    def forward(self, input_ids1, attention_mask1, input_ids2, attention_mask2, input_ids3, attention_mask3, input_ids4, attention_mask4, labels=None):\n",
    "        outputs = self.base_model(input_ids1, attention_mask=attention_mask1)\n",
    "        pooled_output = outputs[1]\n",
    "        logits = self.classifier(pooled_output)\n",
    "\n",
    "        loss = None\n",
    "        if labels is not None:\n",
    "            loss = self.cross_entropy_loss(logits, labels)\n",
    "        \n",
    "        return (loss, logits) if loss is not None else logits\n",
    "    \n",
    "set_seed(data_seed)\n",
    "model = Sft_Model(base_model, num_cls)\n",
    "\n",
    "# Define training arguments and trainer\n",
    "training_args = TrainingArguments(\n",
    "    overwrite_output_dir=True,\n",
    "    output_dir=f'./results/yelp/sft/{N}-shot-{data_seed}',\n",
    "    eval_strategy=\"steps\",\n",
    "    save_strategy=\"steps\",\n",
    "    logging_strategy=\"steps\",\n",
    "    logging_steps=0.1,\n",
    "    save_steps=0.1,\n",
    "    learning_rate=5e-5,\n",
    "    per_device_train_batch_size=128,\n",
    "    per_device_eval_batch_size=128,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=True,\n",
    "    metric_for_best_model=\"eval_loss\",  # Choose model based on validation loss\n",
    "    greater_is_better=False,  # Lower validation loss is better\n",
    "    save_total_limit=1,\n",
    ")\n",
    "\n",
    "\n",
    "trainer = Trainer(\n",
    "    model=model,\n",
    "    args=training_args,\n",
    "    train_dataset=paired_train_dataset,\n",
    "    eval_dataset=paired_validation_dataset,\n",
    "    tokenizer=tokenizer,\n",
    "    data_collator=paired_data_collator,\n",
    "    compute_metrics=compute_metrics,\n",
    ")\n",
    "\n",
    "# Train the model\n",
    "trainer.train()\n",
    "\n",
    "results = {}\n",
    "train_results = trainer.evaluate(paired_train_dataset)\n",
    "print(f'train: {train_results}')\n",
    "results['train'] = train_results\n",
    "\n",
    "valid_results = trainer.evaluate(paired_validation_dataset)\n",
    "print(f'validation: {valid_results}')\n",
    "results['valid'] = valid_results\n",
    "\n",
    "# Evaluate on the test set\n",
    "test_results_ID_90 = trainer.evaluate(paired_test_dataset_ID_90)\n",
    "print(f'ID 90: {test_results_ID_90}')\n",
    "results['ID 90'] = test_results_ID_90\n",
    "\n",
    "test_results_OOD_change_70 = trainer.evaluate(paired_test_dataset_OOD_change_70)\n",
    "print(f'OOD change 70: {test_results_OOD_change_70}')\n",
    "results['OOD change 70'] = test_results_OOD_change_70\n",
    "\n",
    "test_results_OOD_balanced_50 = trainer.evaluate(paired_test_dataset_OOD_balanced_50)\n",
    "print(f'OOD balanced 50: {test_results_OOD_balanced_50}')\n",
    "results['OOD balanced 50'] = test_results_OOD_balanced_50\n",
    "\n",
    "test_results_OOD_change_30 = trainer.evaluate(paired_test_dataset_OOD_change_30)\n",
    "print(f'OOD change 30: {test_results_OOD_change_30}')\n",
    "results['OOD change 30'] = test_results_OOD_change_30\n",
    "\n",
    "test_results_OOD_flip_10 = trainer.evaluate(paired_test_dataset_OOD_flip_10)\n",
    "print(f'OOD flip: {test_results_OOD_flip_10}')\n",
    "results['OOD flip'] = test_results_OOD_flip_10\n",
    "\n",
    "test_results_OOD_original_0 = trainer.evaluate(paired_test_dataset_OOD_original_0)\n",
    "print(f'OOD original: {test_results_OOD_original_0}')\n",
    "results['OOD original'] = test_results_OOD_original_0\n",
    "\n",
    "\n",
    "# Manually save the results\n",
    "with open(f'./results/yelp/sft/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "plot_training_metrics(trainer)\n",
    "\n",
    "# Save the full model after training is complete\n",
    "# torch.save(model.state_dict(), f'./results/yelp/sft/{N}-shot-{data_seed}/sft_model.pt')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 3. Causal SFT"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Initialise Causal SFT model \"bert-base-uncased\" (unfreezed all layers) with a linear head!\n",
      "Total Parameters: 252520708\n",
      "Trainable Parameters: 143038468\n",
      "Percentage of Trainable Parameters: 56.6443%\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='630' max='630' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [630/630 04:26, Epoch 10/10]\n",
       "    </div>\n",
       "    <table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       " <tr style=\"text-align: left;\">\n",
       "      <th>Step</th>\n",
       "      <th>Training Loss</th>\n",
       "      <th>Validation Loss</th>\n",
       "      <th>F1</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <td>63</td>\n",
       "      <td>0.806700</td>\n",
       "      <td>0.757010</td>\n",
       "      <td>0.926496</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>126</td>\n",
       "      <td>0.355200</td>\n",
       "      <td>0.684670</td>\n",
       "      <td>0.922379</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>189</td>\n",
       "      <td>0.146000</td>\n",
       "      <td>0.608880</td>\n",
       "      <td>0.937498</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>252</td>\n",
       "      <td>0.070600</td>\n",
       "      <td>0.647795</td>\n",
       "      <td>0.933976</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>315</td>\n",
       "      <td>0.031000</td>\n",
       "      <td>0.695449</td>\n",
       "      <td>0.930999</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>378</td>\n",
       "      <td>0.003600</td>\n",
       "      <td>0.659379</td>\n",
       "      <td>0.933500</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>441</td>\n",
       "      <td>-0.011800</td>\n",
       "      <td>0.800130</td>\n",
       "      <td>0.932488</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>504</td>\n",
       "      <td>-0.018700</td>\n",
       "      <td>0.770539</td>\n",
       "      <td>0.931985</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>567</td>\n",
       "      <td>-0.022800</td>\n",
       "      <td>0.751498</td>\n",
       "      <td>0.930999</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>630</td>\n",
       "      <td>-0.028000</td>\n",
       "      <td>0.642168</td>\n",
       "      <td>0.930998</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table><p>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='271' max='63' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [63/63 00:46]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train: {'eval_loss': 0.35910359025001526, 'eval_f1': 0.9941249793456306, 'eval_runtime': 10.6693, 'eval_samples_per_second': 749.815, 'eval_steps_per_second': 5.905, 'epoch': 10.0}\n",
      "validation: {'eval_loss': 0.6300422549247742, 'eval_f1': 0.9374981093178069, 'eval_runtime': 3.0072, 'eval_samples_per_second': 665.073, 'eval_steps_per_second': 5.321, 'epoch': 10.0}\n",
      "ID 90: {'eval_loss': 0.6570214033126831, 'eval_f1': 0.9322492843830663, 'eval_runtime': 5.5586, 'eval_samples_per_second': 719.606, 'eval_steps_per_second': 5.757, 'epoch': 10.0}\n",
      "OOD change 70: {'eval_loss': 1.0354777574539185, 'eval_f1': 0.8539894507378158, 'eval_runtime': 5.5817, 'eval_samples_per_second': 716.632, 'eval_steps_per_second': 5.733, 'epoch': 10.0}\n",
      "OOD balanced 50: {'eval_loss': 1.4247701168060303, 'eval_f1': 0.7729304599533606, 'eval_runtime': 5.5507, 'eval_samples_per_second': 720.634, 'eval_steps_per_second': 5.765, 'epoch': 10.0}\n",
      "OOD change 30: {'eval_loss': 1.793558120727539, 'eval_f1': 0.6926239566112078, 'eval_runtime': 5.5975, 'eval_samples_per_second': 714.601, 'eval_steps_per_second': 5.717, 'epoch': 10.0}\n",
      "OOD flip: {'eval_loss': 2.230099678039551, 'eval_f1': 0.6072642603722884, 'eval_runtime': 5.5692, 'eval_samples_per_second': 718.23, 'eval_steps_per_second': 5.746, 'epoch': 10.0}\n",
      "OOD original: {'eval_loss': 1.0534745454788208, 'eval_f1': 0.8258405687360557, 'eval_runtime': 5.5375, 'eval_samples_per_second': 722.35, 'eval_steps_per_second': 5.779, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA04AAAINCAYAAAAJGy/3AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAAB79ElEQVR4nO3dd3hUZfrG8e/MpHcgpAChhN57BwVBEBELFhQVUWQXBRVZy6q/te0q6q7IKiuWVbEi9hVFKSodpHekQygJgUAK6Zk5vz9OMiGUJECSk0zuz3Wdi8yZMzPPOBJy533f57UZhmEgIiIiIiIi52W3ugAREREREZHKTsFJRERERESkBApOIiIiIiIiJVBwEhERERERKYGCk4iIiIiISAkUnEREREREREqg4CQiIiIiIlICBScREREREZESeFldQEVzuVwcOXKE4OBgbDab1eWIiIiIiIhFDMMgLS2NOnXqYLcXP6ZU7YLTkSNHiImJsboMERERERGpJA4ePEi9evWKvabaBafg4GDA/I8TEhJicTUiIiIiImKV1NRUYmJi3BmhONUuOBVMzwsJCVFwEhERERGRUi3hUXMIERERERGREig4iYiIiIiIlEDBSUREREREpAQKTiIiIiIiIiVQcBIRERERESmBgpOIiIiIiEgJFJxERERERERKoOAkIiIiIiJSAgUnERERERGREig4iYiIiIiIlEDBSUREREREpAQKTiIiIiIiIiVQcBIRERERESmBgpOIiIiIiEgJFJxERERERERKoOAkIiIiIiJSAgUnC2XmOPlh0xHWxZ20uhQRERERESmG5cHpzTffpFGjRvj5+dG5c2eWLFlS7PWffvop7du3JyAggOjoaO6++26SkpIqqNqy9dqCnUz4bD3vL91ndSkiIiIiIlIMS4PTrFmzmDhxIk899RTr16+nb9++DBkyhLi4uHNev3TpUkaNGsWYMWPYunUrX375JatXr+bee++t4MrLxtC20QD8sj2RjJw8i6sREREREZHzsTQ4TZkyhTFjxnDvvffSsmVLpk6dSkxMDNOnTz/n9StXrqRhw4Y8+OCDNGrUiD59+vDnP/+ZNWvWVHDlZaNdvVDq1wwgM9fJgu2JVpcjIiIiIiLnYVlwysnJYe3atQwaNKjI+UGDBrF8+fJzPqZXr14cOnSIOXPmYBgGR48e5auvvmLo0KHn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9cPPz49PPvmEpKQkbrvtNurVq0dAQABt27Zl5syZRZ63X79+TJw40X27YcOGvPjii9xzzz0EBwdTv3593nnnHff9+/fvx2azsWHDBgAWLlyIzWbjl19+oUuXLgQEBNCrV68ioQ7gH//4BxEREQQHB3Pvvffy17/+9ZJGkrKzs3nwwQeJiIjAz8+PPn36sHr1avf9J0+e5Pbbb6d27dr4+/vTtGlTPvjgAwBycnKYMGEC0dHR+Pn50bBhQyZPnnzRtYhIxTKn+wUxtF00kwY1551RXVj0aH+2PjeY78b35uUb23J374b0alyLmoE+GAbsO57Oz1sT+Pcvu7jv03Vc8eoiWj/zM8PeWMqjX27kv0v2snTXcY6lZVv99kRExEIacSpBZq6TVk/PteS1tz0/mACfsvmIHn/8cV599VU++OADfH19ycrKonPnzjz++OOEhITw448/cueddxIbG0v37t3P+zyvvvoqf//733nyySf56quvuO+++7jsssto0aLFeR/z1FNP8eqrr1K7dm3GjRvHPffcw7JlywD49NNPeeGFF3jzzTfp3bs3n3/+Oa+++iqNGjW66Pf62GOP8fXXX/Phhx/SoEEDXnnlFQYPHszu3bupWbMmf/vb39i2bRs//fQT4eHh7N69m8zMTABef/11vv/+e7744gvq16/PwYMHOXjw4EXXIiKVQ4CPFx1iwugQE+Y+ZxgGx05lF+nstyMhjZ1H08jKdbH5cAqbDxed7lcr0KfIVL/mUSE0iwwqs+/VIiJSeek7fTUxceJEhg8fXuTcI4884v76gQce4Oeff+bLL78sNjhdffXV3H///YAZxl577TUWLlxYbHB64YUXuPzyywH461//ytChQ8nKysLPz4833niDMWPGcPfddwPw9NNPM2/ePE6dOnVR7zM9PZ3p06czY8YMhgwZAsC7777L/Pnzee+993j00UeJi4ujY8eOdOnSBTBH0grExcXRtGlT+vTpg81mo0GDBhdVh4hUfjabjYhgPyKC/ejbtHC6n9NlcCApvUiY2nE0jf1J6SSl57B8TxLL9ySd9jxQv2aAO0gVTPdrqOl+IiIeRcGpBP7eDrY9P9iy1y4rBSGhgNPp5KWXXmLWrFkcPnyY7OxssrOzCQwMLPZ52rUrbHVVMCUwMTGx1I+Jjo4GIDExkfr167Njxw53ECvQrVs3fv31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06dPx+FwsGHDBndIGz9+PDk5OSxevJjAwEC2bdvmDnkiIiJS9VkZ6ApCW1bu+QPZuf4sCF7ZxQSzs88VBryC1v1QPoHt6/t60VnBSSraPffcwz//+U8WLlxI//79AXOa3vDhw6lRowY1atTgkUcecV//wAMP8PPPP/Pll1+WKjgtWLCA7du3s3//furVqwfAiy++yJAhQ4pc93//93/urxs2bMhf/vIXZs2axWOPPYa/vz9BQUF4eXkVOzXv008/JTMzk48++si9xmratGkMGzaMl19+2T0CVKNGDaZNm4bD4aBFixYMHTqUX3755aKC04IFC9i0aRP79u0jJiYGgI8//pjWrVuzevVqunbtSlxcHI8++igtWrQAoGnTpu7Hx8XFceONN9K2bVsAYmNjL7gGERERkXOxKrS5XOYauNKOjBUdeSv5mrLubFneFJxK4h1gjvxY9dql1KJFC3r16sX7779P//792bNnD0uWLGHevHkAOJ1OXnrpJWbNmsXhw4fJzs4mOzu71M0ftm/fTv369d2hCaBnz55nXffVV18xdepUdu/ezalTp8jLyytxF+ZzvVb79u2L1Na7d29cLhc7duxwB6fWrVvjcBR+A4mOjmbz5s0X9Fqnv2ZMTIw7NAG0atWKsLAwtm/fTteuXZk0aRL33nsvH3/8MQMHDuTmm2+mcePGADz44IPcd999zJs3j4EDB3LjjTfSrl27i6pFREREpDKw2ws7RQpolVdJbDZzupwVxwW2uRwzZgxff/01qampfPDBBzRo0IABAwYA8Oqrr/Laa6/x2GOP8euvv7JhwwYGDx5MTk5OqZ7bMIyzzp3ZhnPlypXceuutDBkyhB9++IH169fz1FNPlfo1Tn+t87X4PP38mWuZbDYbLpfrzIdc0muefv7ZZ59l69atDB06lF9//ZVWrVrx7bffAnDvvfeyd+9e7rzzTjZv3kyXLl144403LqoWEREREal8FJw8yC233ILD4eCzzz7jww8/5O6773b/0L9kyRKuu+467rjjDtq3b09sbCy7du0q9XO3atWKuLg4jhwpHH1bsWJFkWuWLVtGgwYNeOqpp+jSpQtNmzblwIEDRa7x8fHB6Sx+bmyrVq3YsGED6emF260vW7YMu91Os2bNSl3zhSh4fwcPHnSf27ZtGykpKbRs2dJ9rlmzZjz88MPMmzeP4cOH88EHH7jvi4mJYdy4cXzzzTf85S9/4d133y2XWkVERESk4ik4eZCgoCBGjBjBk08+yZEjRxg9erT7viZNmjB//nyWL1/O9u3b+fOf/0xCQkKpn3vgwIE0b96cUaNGsXHjRpYsWcJTTz1V5JomTZoQFxfH559/zp49e3j99dfdIzIFGjZsyL59+9iwYQPHjx8nO/vsXc9vv/12/Pz8uOuuu9iyZQu//fYbDzzwAHfeead7mt7FcjqdbNiwocixbds2Bg4cSLt27bj99ttZt24dq1atYtSoUVx++eV06dKFzMxMJkyYwMKFCzlw4ADLli1j9erV7lA1ceJE5s6dy759+1i3bh2//vprkcAlIiIiIlWbgpOHGTNmDCdPnmTgwIHUr1/fff5vf/sbnTp1YvDgwfTr14+oqCh36+/SsNvtfPvtt2RnZ9OtWzfuvfdeXnjhhSLXXHfddTz88MNMmDCBDh06sHz5cv72t78VuebGG2/kqquuon///tSuXZuZM2ee9VoBAQHMnTuXEydO0LVrV2666SYGDBjAtGnTLuw/xjmcOnWKjh07FjmuvvpqbDYb3333HTVq1OCyyy5j4MCBxMbGMmvWLAAcDgdJSUmMGjWKZs2accsttzBkyBB3e3Wn08n48eNp2bIlV111Fc2bN+fNN9+85HpFREREpHKwGedavOLBUlNTCQ0NJSUl5aymBVlZWezbt49GjRrh5+dnUYXi6fT/mYiIiEjlUFw2OJNGnEREREREREqg4CQiIiIiIlICBScREREREZESKDiJiIiIiIiUQMHpHKpZvwypYPr/S0RERKTqUXA6jbe3NwAZGRkWVyKerOD/r4L/30RERESk8vOyuoDKxOFwEBYWRmJiImDuJ2Sz2SyuSjyFYRhkZGSQmJhIWFgYDofD6pJEREREpJQUnM4QFRUF4A5PImUtLCzM/f+ZiIiIiFQNCk5nsNlsREdHExERQW5urtXliIfx9vbWSJOIiIhIFaTgdB4Oh0M/4IqIiIiICKDmECIiIiIiIiVScBIRERERESmBgpOIiIiIiEgJqt0ap4LNR1NTUy2uRERERERErFSQCQoyQnGqXXBKS0sDICYmxuJKRERERESkMkhLSyM0NLTYa2xGaeKVB3G5XBw5coTg4GBtbnsJUlNTiYmJ4eDBg4SEhFhdjpQzfd7Viz7v6kWfd/Wjz7x60eddPMMwSEtLo06dOtjtxa9iqnYjTna7nXr16lldhscICQnRX8JqRJ939aLPu3rR51396DOvXvR5n19JI00F1BxCRERERESkBApOIiIiIiIiJVBwkovi6+vLM888g6+vr9WlSAXQ51296POuXvR5Vz/6zKsXfd5lp9o1hxAREREREblQGnESEREREREpgYKTiIiIiIhICRScRERERERESqDgJCIiIiIiUgIFJ3FbvHgxw4YNo06dOthsNr777rsi9xuGwbPPPkudOnXw9/enX79+bN26tcg12dnZPPDAA4SHhxMYGMi1117LoUOHKvBdSGlNnjyZrl27EhwcTEREBNdffz07duwoco0+c88xffp02rVr594AsWfPnvz000/u+/VZe7bJkydjs9mYOHGi+5w+c8/x7LPPYrPZihxRUVHu+/VZe6bDhw9zxx13UKtWLQICAujQoQNr165136/PvewpOIlbeno67du3Z9q0aee8/5VXXmHKlClMmzaN1atXExUVxZVXXklaWpr7mokTJ/Ltt9/y+eefs3TpUk6dOsU111yD0+msqLchpbRo0SLGjx/PypUrmT9/Pnl5eQwaNIj09HT3NfrMPUe9evV46aWXWLNmDWvWrOGKK67guuuuc/8jqs/ac61evZp33nmHdu3aFTmvz9yztG7dmvj4ePexefNm9336rD3PyZMn6d27N97e3vz0009s27aNV199lbCwMPc1+tzLgSFyDoDx7bffum+7XC4jKirKeOmll9znsrKyjNDQUOOtt94yDMMwkpOTDW9vb+Pzzz93X3P48GHDbrcbP//8c4XVLhcnMTHRAIxFixYZhqHPvDqoUaOG8d///leftQdLS0szmjZtasyfP9+4/PLLjYceesgwDP399jTPPPOM0b59+3Pep8/aMz3++ONGnz59znu/PvfyoREnKZV9+/aRkJDAoEGD3Od8fX25/PLLWb58OQBr164lNze3yDV16tShTZs27muk8kpJSQGgZs2agD5zT+Z0Ovn8889JT0+nZ8+e+qw92Pjx4xk6dCgDBw4scl6fuefZtWsXderUoVGjRtx6663s3bsX0Gftqb7//nu6dOnCzTffTEREBB07duTdd99136/PvXwoOEmpJCQkABAZGVnkfGRkpPu+hIQEfHx8qFGjxnmvkcrJMAwmTZpEnz59aNOmDaDP3BNt3ryZoKAgfH19GTduHN9++y2tWrXSZ+2hPv/8c9atW8fkyZPPuk+fuWfp3r07H330EXPnzuXdd98lISGBXr16kZSUpM/aQ+3du5fp06fTtGlT5s6dy7hx43jwwQf56KOPAP0dLy9eVhcgVYvNZity2zCMs86dqTTXiLUmTJjApk2bWLp06Vn36TP3HM2bN2fDhg0kJyfz9ddfc9ddd7Fo0SL3/fqsPcfBgwd56KGHmDdvHn5+fue9Tp+5ZxgyZIj767Zt29KzZ08aN27Mhx9+SI8ePQB91p7G5XLRpUsXXnzxRQA6duzI1q1bmT59OqNGjXJfp8+9bGnESUqloDvPmb+BSExMdP82IyoqipycHE6ePHnea6TyeeCBB/j+++/57bffqFevnvu8PnPP4+PjQ5MmTejSpQuTJ0+mffv2/Pvf/9Zn7YHWrl1LYmIinTt3xsvLCy8vLxYtWsTrr7+Ol5eX+zPTZ+6ZAgMDadu2Lbt27dLfbw8VHR1Nq1atipxr2bIlcXFxgP4NLy8KTlIqjRo1Iioqivnz57vP5eTksGjRInr16gVA586d8fb2LnJNfHw8W7ZscV8jlYdhGEyYMIFvvvmGX3/9lUaNGhW5X5+55zMMg+zsbH3WHmjAgAFs3ryZDRs2uI8uXbpw++23s2HDBmJjY/WZe7Ds7Gy2b99OdHS0/n57qN69e5+1hcjOnTtp0KABoH/Dy40FDSmkkkpLSzPWr19vrF+/3gCMKVOmGOvXrzcOHDhgGIZhvPTSS0ZoaKjxzTffGJs3bzZuu+02Izo62khNTXU/x7hx44x69eoZCxYsMNatW2dcccUVRvv27Y28vDyr3pacx3333WeEhoYaCxcuNOLj491HRkaG+xp95p7jiSeeMBYvXmzs27fP2LRpk/Hkk08adrvdmDdvnmEY+qyrg9O76hmGPnNP8pe//MVYuHChsXfvXmPlypXGNddcYwQHBxv79+83DEOftSdatWqV4eXlZbzwwgvGrl27jE8//dQICAgwPvnkE/c1+tzLnoKTuP32228GcNZx1113GYZhtrZ85plnjKioKMPX19e47LLLjM2bNxd5jszMTGPChAlGzZo1DX9/f+Oaa64x4uLiLHg3UpJzfdaA8cEHH7iv0WfuOe655x6jQYMGho+Pj1G7dm1jwIAB7tBkGPqsq4Mzg5M+c88xYsQIIzo62vD29jbq1KljDB8+3Ni6dav7fn3Wnmn27NlGmzZtDF9fX6NFixbGO++8U+R+fe5lz2YYhmHNWJeIiIiIiEjVoDVOIiIiIiIiJVBwEhERERERKYGCk4iIiIiISAkUnEREREREREqg4CQiIiIiIlICBScREREREZESKDiJiIiIiIiUQMFJRERERESkBApOIiJS5SQmJvLnP/+Z+vXr4+vrS1RUFIMHD2bFihUA2Gw2vvvuO2uLFBERj+JldQEiIiIX6sYbbyQ3N5cPP/yQ2NhYjh49yi+//MKJEyesLk1ERDyURpxERKRKSU5OZunSpbz88sv079+fBg0a0K1bN5544gmGDh1Kw4YNAbjhhhuw2Wzu2wCzZ8+mc+fO+Pn5ERsby3PPPUdeXp77fpvNxvTp0xkyZAj+/v40atSIL7/80n1/Tk4OEyZMIDo6Gj8/Pxo2bMjkyZMr6q2LiIiFFJxERKRKCQoKIigoiO+++47s7Oyz7l+9ejUAH3zwAfHx8e7bc+fO5Y477uDBBx9k27ZtvP3228yYMYMXXnihyOP/9re/ceONN7Jx40buuOMObrvtNrZv3w7A66+/zvfff88XX3zBjh07+OSTT4oEMxER8Vw2wzAMq4sQERG5EF9//TVjx44lMzOTTp06cfnll3PrrbfSrl07wBw5+vbbb7n++uvdj7nssssYMmQITzzxhPvcJ598wmOPPcaRI0fcjxs3bhzTp093X9OjRw86derEm2++yYMPPsjWrVtZsGABNputYt6siIhUChpxEhGRKufGG2/kyJEjfP/99wwePJiFCxfSqVMnZsyYcd7HrF27lueff949YhUUFMTYsWOJj48nIyPDfV3Pnj2LPK5nz57uEafRo0ezYcMGmjdvzoMPPsi8efPK5f2JiEjlo+AkIiJVkp+fH1deeSVPP/00y5cvZ/To0TzzzDPnvd7lcvHcc8+xYcMG97F582Z27dqFn59fsa9VMLrUqVMn9u3bx9///ncyMzO55ZZbuOmmm8r0fYmISOWk4CQiIh6hVatWpKenA+Dt7Y3T6Sxyf6dOndixYwdNmjQ567DbC/85XLlyZZHHrVy5khYtWrhvh4SEMGLECN59911mzZrF119/rW5+IiLVgNqRi4hIlZKUlMTNN9/MPffcQ7t27QgODmbNmjW88sorXHfddQA0bNiQX375hd69e+Pr60uNGjV4+umnueaaa4iJieHmm2/GbrezadMmNm/ezD/+8Q/383/55Zd06dKFPn368Omnn7Jq1Sree+89AF577TWio6Pp0KEDdrudL7/8kqioKMLCwqz4TyEiIhVIwUlERKqUoKAgunfvzmuvvcaePXvIzc0lJiaGsWPH8uSTTwLw6quvMmnSJN59913q1q3L/v37GTx4MD/88APPP/88r7zyCt7e3rRo0YJ77723yPM/99xzfP7559x///1ERUXx6aef0qpVK/drv/zyy+zatQuHw0HXrl2ZM2dOkRErERHxTOqqJyIiku9c3fhERERAa5xERERERERKpOAkIiIiIiJSAq1xEhERyafZ6yIicj4acRIRERERESmBgpOIiIiIiEgJFJxERERERERKoOAkIiIiIiJSAgUnERERERGREig4iYiIiIiIlEDBSUREREREpAQKTiIiIiIiIiVQcBIRERERESnB/wOmGQGcJjWRKAAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "class CausalSft_Model(nn.Module):\n",
    "    def __init__(self, base_model, num_labels, max_words):\n",
    "        super().__init__()\n",
    "\n",
    "        self.max_words = max_words\n",
    "        self.base_model = copy.deepcopy(base_model)\n",
    "        self.sft_model = copy.deepcopy(base_model)\n",
    "        self.sft_classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "        self.r_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 2, self.base_model.config.hidden_size // 2),\n",
    "        )\n",
    "        self.c_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "        )\n",
    "        self.num_patches = 10\n",
    "        self.patches_size = self.max_words // self.num_patches\n",
    "        self.patch_extractor = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches, self.base_model.config.hidden_size * self.num_patches // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches // 2, self.base_model.config.hidden_size),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size),\n",
    "        )\n",
    "        self.classifier = nn.Linear(self.base_model.config.hidden_size + self.base_model.config.hidden_size // 4, num_labels)\n",
    "\n",
    "        self.cross_entropy_loss = nn.CrossEntropyLoss()\n",
    "        self.mse_loss = nn.MSELoss()\n",
    "\n",
    "        for param in self.base_model.parameters():\n",
    "            param.requires_grad = False\n",
    "\n",
    "        print(f'Initialise Causal SFT model \"{model_name}\" (unfreezed all layers) with a linear head!')\n",
    "        count_parameters(self)\n",
    "\n",
    "    def entropy_maximization(self, feature):\n",
    "        p = F.softmax(feature, dim=-1)\n",
    "        log_p = F.log_softmax(feature, dim=-1)\n",
    "        entropy = -torch.mean(torch.mean(p * log_p, dim=-1))  # Maximize entropy\n",
    "        return -entropy  # Minimize negative entropy\n",
    "\n",
    "    def forward(self, input_ids1, attention_mask1, input_ids2, attention_mask2, input_ids3, attention_mask3, input_ids4, attention_mask4, labels=None):\n",
    "        B, _ = input_ids1.size()\n",
    "        # avoid unwanted computational graph\n",
    "        outputs_sft2 = self.sft_model(input_ids2, attention_mask=attention_mask2)[1]\n",
    "        outputs_sft2_logits = self.sft_classifier(outputs_sft2)\n",
    "        # pull out R0 and R1 using another data point\n",
    "        outputs_base3 = self.base_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        outputs_sft3 = self.sft_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        # optimise C from R0 and R1\n",
    "        base_C3= self.r_2_c(outputs_base3)[:, :self.base_model.config.hidden_size//4]\n",
    "        sft_C3 = self.r_2_c(outputs_sft3)[:, :self.base_model.config.hidden_size//4]\n",
    "        # gather C \n",
    "        outputs_sft = self.sft_model(input_ids1, attention_mask=attention_mask1)[1]\n",
    "        sft_C = self.r_2_c(outputs_sft)[:, :self.base_model.config.hidden_size//4]\n",
    "        # extract non-contextual embeddings\n",
    "        with torch.no_grad():\n",
    "            embeddings_1 = self.sft_model.embeddings(input_ids1, attention_mask1)[:, :self.max_words]\n",
    "            embeddings_2 = self.sft_model.embeddings(input_ids2, attention_mask2)[:, :self.max_words]\n",
    "            embeddings_4 = self.sft_model.embeddings(input_ids4, attention_mask4)[:, :self.max_words]\n",
    "            \n",
    "        if self.training:\n",
    "            input = embeddings_2\n",
    "            patches = torch.sum(input.view(B, self.num_patches, self.patches_size, -1), dim=2)\n",
    "            Ai_samples = self.patch_extractor(patches.view(B, -1))\n",
    "            Ai_samples = Ai_samples[torch.randperm(B).to(input_ids1.device), :]\n",
    "            Ai_Z1 = torch.cat([Ai_samples, self.c_2_c(sft_C)], dim=-1)\n",
    "            logits = self.classifier(Ai_Z1)\n",
    "        else:\n",
    "            num_samples = 20\n",
    "            logits = 0\n",
    "            input = embeddings_1\n",
    "            patches = torch.sum(input.view(B, self.num_patches, self.patches_size, -1), dim=2)\n",
    "            Ai_samples = self.patch_extractor(patches.view(B, -1))\n",
    "            for _ in range(num_samples):\n",
    "                Ai_samples = Ai_samples[torch.randperm(B).to(input_ids1.device), :]\n",
    "                Ai_Z1 = torch.cat([Ai_samples, self.c_2_c(sft_C)], dim=-1)\n",
    "                logits += F.softmax(self.classifier(Ai_Z1), dim=-1)\n",
    "            logits = logits / num_samples\n",
    "\n",
    "        loss = None\n",
    "        if labels is not None:\n",
    "            loss_sft = self.cross_entropy_loss(outputs_sft2_logits, labels) \n",
    "            loss_identify = self.mse_loss(base_C3, sft_C3) + self.entropy_maximization(sft_C3) + self.entropy_maximization(base_C3)\n",
    "            loss_cls = self.cross_entropy_loss(logits, labels) \n",
    "            \n",
    "            loss = loss_sft + loss_identify + loss_cls \n",
    "        \n",
    "        return (loss, logits) if loss is not None else logits\n",
    "    \n",
    "    def forward2(self, input_ids, attention_mask):\n",
    "        B, _ = input_ids.size()\n",
    "        R0 = self.base_model(input_ids, attention_mask=attention_mask)[1]\n",
    "        R1 = self.sft_model(input_ids, attention_mask=attention_mask)[1]\n",
    "        C = self.r_2_c(R1)[:, :self.base_model.config.hidden_size//4]\n",
    "        input = self.sft_model.embeddings(input_ids, attention_mask=attention_mask)[:, :self.max_words]\n",
    "        patches = torch.sum(input.view(B, self.num_patches, self.patches_size, -1), dim=2)\n",
    "        Ai_samples = self.patch_extractor(patches.view(B, -1))\n",
    "        num_samples = 20\n",
    "        Ai_Z1_ = 0\n",
    "        for _ in range(num_samples):\n",
    "            Ai_samples = Ai_samples[torch.randperm(B).to(input_ids.device), :]\n",
    "            Ai_Z1 = torch.cat([Ai_samples, self.c_2_c(C)], dim=-1)\n",
    "            Ai_Z1_ += Ai_Z1\n",
    "        Ai_Z1_ = Ai_Z1_ / num_samples\n",
    "        return (R0, R1, C, Ai_samples, Ai_Z1_)\n",
    "    \n",
    "set_seed(data_seed)\n",
    "model = CausalSft_Model(base_model, num_cls, max_words)\n",
    "\n",
    "# Define training arguments and trainer\n",
    "training_args = TrainingArguments(\n",
    "    overwrite_output_dir=True,\n",
    "    output_dir=f'./results/yelp/causal-sft/{N}-shot-{data_seed}',\n",
    "    eval_strategy=\"steps\",\n",
    "    save_strategy=\"steps\",\n",
    "    logging_strategy=\"steps\",\n",
    "    logging_steps=0.1,\n",
    "    save_steps=0.1,\n",
    "    learning_rate=5e-5,\n",
    "    per_device_train_batch_size=128,\n",
    "    per_device_eval_batch_size=128,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=True,\n",
    "    metric_for_best_model=\"eval_loss\",  # Choose model based on validation loss\n",
    "    greater_is_better=False,  # Lower validation loss is better\n",
    "    save_total_limit=1,\n",
    ")\n",
    "\n",
    "\n",
    "trainer = Trainer(\n",
    "    model=model,\n",
    "    args=training_args,\n",
    "    train_dataset=paired_train_dataset,\n",
    "    eval_dataset=paired_validation_dataset,\n",
    "    tokenizer=tokenizer,\n",
    "    data_collator=paired_data_collator,\n",
    "    compute_metrics=compute_metrics,\n",
    ")\n",
    "\n",
    "# Train the model\n",
    "trainer.train()\n",
    "\n",
    "results = {}\n",
    "train_results = trainer.evaluate(paired_train_dataset)\n",
    "print(f'train: {train_results}')\n",
    "results['train'] = train_results\n",
    "\n",
    "valid_results = trainer.evaluate(paired_validation_dataset)\n",
    "print(f'validation: {valid_results}')\n",
    "results['valid'] = valid_results\n",
    "\n",
    "# Evaluate on the test set\n",
    "test_results_ID_90 = trainer.evaluate(paired_test_dataset_ID_90)\n",
    "print(f'ID 90: {test_results_ID_90}')\n",
    "results['ID 90'] = test_results_ID_90\n",
    "\n",
    "test_results_OOD_change_70 = trainer.evaluate(paired_test_dataset_OOD_change_70)\n",
    "print(f'OOD change 70: {test_results_OOD_change_70}')\n",
    "results['OOD change 70'] = test_results_OOD_change_70\n",
    "\n",
    "test_results_OOD_balanced_50 = trainer.evaluate(paired_test_dataset_OOD_balanced_50)\n",
    "print(f'OOD balanced 50: {test_results_OOD_balanced_50}')\n",
    "results['OOD balanced 50'] = test_results_OOD_balanced_50\n",
    "\n",
    "test_results_OOD_change_30 = trainer.evaluate(paired_test_dataset_OOD_change_30)\n",
    "print(f'OOD change 30: {test_results_OOD_change_30}')\n",
    "results['OOD change 30'] = test_results_OOD_change_30\n",
    "\n",
    "test_results_OOD_flip_10 = trainer.evaluate(paired_test_dataset_OOD_flip_10)\n",
    "print(f'OOD flip: {test_results_OOD_flip_10}')\n",
    "results['OOD flip'] = test_results_OOD_flip_10\n",
    "\n",
    "test_results_OOD_original_0 = trainer.evaluate(paired_test_dataset_OOD_original_0)\n",
    "print(f'OOD original: {test_results_OOD_original_0}')\n",
    "results['OOD original'] = test_results_OOD_original_0\n",
    "\n",
    "# Manually save the results\n",
    "with open(f'./results/yelp/causal-sft/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "plot_training_metrics(trainer)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 4. Causal SFT - No Front Door"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Initialise Causal SFT model \"bert-base-uncased\" (unfreezed all layers) with a linear head!\n",
      "Total Parameters: 252520708\n",
      "Trainable Parameters: 143038468\n",
      "Percentage of Trainable Parameters: 56.6443%\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='630' max='630' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [630/630 04:41, Epoch 10/10]\n",
       "    </div>\n",
       "    <table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       " <tr style=\"text-align: left;\">\n",
       "      <th>Step</th>\n",
       "      <th>Training Loss</th>\n",
       "      <th>Validation Loss</th>\n",
       "      <th>F1</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <td>63</td>\n",
       "      <td>0.554300</td>\n",
       "      <td>0.477456</td>\n",
       "      <td>0.897999</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>126</td>\n",
       "      <td>0.307800</td>\n",
       "      <td>0.405266</td>\n",
       "      <td>0.926427</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>189</td>\n",
       "      <td>0.119900</td>\n",
       "      <td>0.424851</td>\n",
       "      <td>0.930496</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>252</td>\n",
       "      <td>0.027800</td>\n",
       "      <td>0.505521</td>\n",
       "      <td>0.928994</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>315</td>\n",
       "      <td>-0.002100</td>\n",
       "      <td>0.553772</td>\n",
       "      <td>0.930499</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>378</td>\n",
       "      <td>-0.026400</td>\n",
       "      <td>0.585621</td>\n",
       "      <td>0.928433</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>441</td>\n",
       "      <td>-0.036300</td>\n",
       "      <td>0.691052</td>\n",
       "      <td>0.930489</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>504</td>\n",
       "      <td>-0.038400</td>\n",
       "      <td>0.641708</td>\n",
       "      <td>0.936499</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>567</td>\n",
       "      <td>-0.035100</td>\n",
       "      <td>0.624195</td>\n",
       "      <td>0.936995</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>630</td>\n",
       "      <td>-0.042000</td>\n",
       "      <td>0.555050</td>\n",
       "      <td>0.934996</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table><p>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='271' max='63' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [63/63 00:46]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train: {'eval_loss': 0.16171099245548248, 'eval_f1': 0.9748688644688644, 'eval_runtime': 10.632, 'eval_samples_per_second': 752.445, 'eval_steps_per_second': 5.926, 'epoch': 10.0}\n",
      "validation: {'eval_loss': 0.4018169045448303, 'eval_f1': 0.9264269971879598, 'eval_runtime': 2.9933, 'eval_samples_per_second': 668.166, 'eval_steps_per_second': 5.345, 'epoch': 10.0}\n",
      "ID 90: {'eval_loss': 0.4096499979496002, 'eval_f1': 0.9236824542021222, 'eval_runtime': 5.556, 'eval_samples_per_second': 719.946, 'eval_steps_per_second': 5.76, 'epoch': 10.0}\n",
      "OOD change 70: {'eval_loss': 0.9817413091659546, 'eval_f1': 0.8151858432355918, 'eval_runtime': 5.5653, 'eval_samples_per_second': 718.74, 'eval_steps_per_second': 5.75, 'epoch': 10.0}\n",
      "OOD balanced 50: {'eval_loss': 1.5451909303665161, 'eval_f1': 0.7059725859050828, 'eval_runtime': 5.5576, 'eval_samples_per_second': 719.736, 'eval_steps_per_second': 5.758, 'epoch': 10.0}\n",
      "OOD change 30: {'eval_loss': 2.0956859588623047, 'eval_f1': 0.5955628367717754, 'eval_runtime': 5.5058, 'eval_samples_per_second': 726.504, 'eval_steps_per_second': 5.812, 'epoch': 10.0}\n",
      "OOD flip: {'eval_loss': 2.6736156940460205, 'eval_f1': 0.47943105115059703, 'eval_runtime': 5.581, 'eval_samples_per_second': 716.72, 'eval_steps_per_second': 5.734, 'epoch': 10.0}\n",
      "OOD original: {'eval_loss': 0.7840045094490051, 'eval_f1': 0.8347118977432701, 'eval_runtime': 5.5359, 'eval_samples_per_second': 722.563, 'eval_steps_per_second': 5.781, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "image/png": "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EpiUlFToY8LDwwHcx2zZsoWOHTvm2v/Mx0Wxfft2srOz6dq1q3vM09OTjh07EhsbC8D999/PjBkzaNOmDY899liufgbDhg1j7dq1NGnShNGjRzN//vzzrqWkWTbjBDB27FiGDh1Khw4diI6O5oMPPiAuLo7hw4cD5mzR3r17mTZtWq7jPv74Yzp16lTiHURERESkHDl2BBLWQUQn8PSxuhq5AL6eDjaN72vJ6xYXf3//XI8nTpzIpEmTmDx5Mi1btsTf358xY8aQlZVV4HnObCphs9lwuVyFPubkLTCnH5PfrTLn6+SxBd1+069fP3bv3s3PP//MggUL6NWrFyNGjOD111+nXbt27Ny5k3nz5rFgwQIGDRpE7969+fbbb8+7ppJiaXAaPHgwhw4dYvz48e5WinPnznVP0SUkJORJ4SkpKcyePZs333zTipJFRESkLNqzCr65DVL3gE8wNOsPrQZDZDTYLb+lW4rIZrMV2yVzZcWSJUu47rrruPXWWwEzyGzdupWmTZuWah1NmjTh77//ZujQoe6xlStXnvf5GjZsiJeXF3/++Sc333wzYHYRXLlyZa5GFzVq1GDYsGEMGzaM7t278+ijj/L6668DEBQUxODBgxk8eDA33HADV1xxBcnJyUXuMljSLP8T+cADD7iv2zzTZ599lmcsODiYjIyMEq5KREREygXDgBUfwS/jwJUNdg84ngKrPze34EhodSO0GgI1GltdrVRiDRs2ZPbs2SxdupQqVarwxhtvkJiYWOrBadSoUdxzzz106NCBLl26MHPmTNavX0/9+vXPeeyZ3fkAmjVrxv3338+jjz5K1apViYyM5NVXXyUjI8N9W80zzzxD+/btad68OZmZmfz000/u9z1p0iTCw8Np06YNdrudWbNmERYWRkhISLG+7+JgeXASEREROS9Z6fDjg7Bhlvm46TVw7TuQuAHWz4BNP0BKHCyZaG7hbcxZqJY3QEBNS0uXyufpp59m586d9O3bFz8/P+6991769+9PSkpKqdZxyy23sGPHDh555BGOHz/OoEGDGDZsGH///fc5jx0yZEiesZ07d/Lyyy/jcrkYOnQoaWlpdOjQgV9//ZUqVaoAZm+DcePGsWvXLnx9fenevTszZswAICAggFdeeYWtW7ficDi4+OKLmTt3LvYyOFNsM4qz72I5kJqaSnBwMCkpKWoU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4hg+fDhgXma3d+9epk2bht1up0WLFrmOr1mzJj4+PnnGpXR5ezh4/IooRny9mvcX7eDmjpHUDCrBRXFFREREREqZpcFp8ODBHDp0iPHjx5OQkECLFi2YO3cuderUASAhIYG4uLhznEXKgitbhtE2MoQ1cUd4I+ZfXh7YyuqSRERERESKjaXrOFlB6ziVnFW7kxk4dRl2G8x7sAdNwgKtLklERERE5KzKxTpOUvG0r1OVK1uG4TJgwrxYq8sRERERESk2Ck5SrB47sSjuwi0HWLK1FBfFFREREREpQQpOUqzqVvfn1s7mPWovalFcEREREakgFJyk2I2+rBGBPh5sTkxjzuo9VpcjIiIiInLBFJyk2FXxP7Uo7uvzt3Asy2lxRSIiIiIiF0bBSUrEbdF1qV3Fl/2pmXykRXFFREREpJxTcJIS4ePp4LErogCYumg7SWnHLa5IREREROT8KThJibmmVTitI0LIyHIyecFWq8sRERERETlvCk5SYmw2G09d2RSAmSvi2bo/zeKKRERERETOj4KTlKiO9arSt3koTpfBy/M2W12OiIiIiMh5UXCSEvf4FVF42G38tjmJpdsOWl2OiIiIiEiRKThJiatfI4BbOkUC8OLcWFxaFFdEREREyhkFJykVo3s1ItDbg437Uvl+7V6ryxERERERKRIFJykV1QK8eeBSc1Hc137dwvFsLYorIiIiIuWHgpOUmju61uWiEF8SUo7z8Z87rS5HRERERKTQFJyk1Ph4Oni0bxMApi7czsGjmRZXJCIiIiJSOApOUqqubV2LlhcFczQzhze1KK6IiIiIlBMKTlKq7HYbT55YFPfrv+PYlnTU4opERERERM5NwUlKXXSDavRuai6K+8ovWhRXRERERMo+BSexxBP9onDYbcRs2s/yHYesLkdEREREpEAKTmKJhjUDuKljBAAvaVFcERERESnjFJzEMmN6NybA24P1e1L4cf0+q8sRERERETkrBSexTPUAb+7v2QCAV3/RorgiIiIiUnYpOIml7uxaj/BgH/YeOcZnS3dZXY6IiIiISL4UnMRSvl4OHuljLor77u/bSE7PsrgiEREREZG8zis4xcfHs2fPHvfjv//+mzFjxvDBBx8UW2FSeVzf9iKahQeRlpnDW79pUVwRERERKXvOKzjdfPPN/PHHHwAkJiZy+eWX8/fff/Pkk08yfvz4Yi1QKj673cb/XWUuivvl8t3sOKBFcUVERESkbDmv4PTPP//QsWNHAL755htatGjB0qVL+frrr/nss8+Ksz6pJLo0rM5lUTXJ0aK4IiIiIlIGnVdwys7OxtvbG4AFCxZw7bXXAhAVFUVCQkLxVSeVyrh+Udht8OvG/azYlWx1OSIiIiIibucVnJo3b857773HkiVLiImJ4YorrgBg3759VKtWrVgLlMqjUWgggy+OBOCFn2MxDC2KKyIiIiJlw3kFp1deeYX333+fnj17ctNNN9G6dWsAfvjhB/clfCLn46HLG+Hn5WBd/BF+Wq/ZSxEREREpG2zGef5Y3+l0kpqaSpUqVdxju3btws/Pj5o1axZbgcUtNTWV4OBgUlJSCAoKsrocycdbv23ljZh/qV3Fl98evgRvD4fVJYmIiIhIBVSUbHBeM07Hjh0jMzPTHZp2797N5MmT2bJlS5kOTVI+3N29HqFB3uw5fIxpS3dbXY6IiIiIyPkFp+uuu45p06YBcOTIETp16sTEiRPp378/U6dOLdYCpfLx8/Lg4ROL4r79+1YOa1FcEREREbHYeQWn1atX0717dwC+/fZbQkND2b17N9OmTeOtt94q1gKlchrYrjZRYYGkHs/h7d+3WV2OiIiIiFRy5xWcMjIyCAwMBGD+/PkMGDAAu91O586d2b1bl1bJhXPYbTx1YlHcL5bvYtfBdIsrEhEREZHK7LyCU8OGDfn++++Jj4/n119/pU+fPgAkJSWp4YIUm+6NanBJ4xpkOw1e/VWL4oqIiIiIdc4rOD3zzDM88sgj1K1bl44dOxIdHQ2Ys09t27Yt1gKlcnvyyqbYbTB3QyKrdh+2uhwRERERqaTOux15YmIiCQkJtG7dGrvdzF9///03QUFBREVFFWuRxUntyMufJ2avZ8aKeNpFhjD7/i7YbDarSxIRERGRCqDE25EDhIWF0bZtW/bt28fevXsB6NixY5kOTVI+jb28Mb6eDlbHHWHeP4lWlyMiIiIildB5BSeXy8X48eMJDg6mTp06REZGEhISwn/+8x9cLldx1yiVXM0gH+7tUR+Al+dtJitHf8ZEREREpHSdV3B66qmneOedd3j55ZdZs2YNq1ev5qWXXuLtt9/m6aefLu4aRbi3R31qBHoTl5zBF8vVuVFEREREStd53eNUq1Yt3nvvPa699tpc4//973954IEH3JfulUW6x6n8mvF3HE/M2UCwryeLH72UYD9Pq0sSERERkXKsxO9xSk5OzvdepqioKJKTk8/nlCLndGOHCJqEBpJyLJt3/thqdTkiIiIiUomcV3Bq3bo177zzTp7xd955h1atWl1wUSL5cdhtjLvSDOyfL91N3KEMiysSERERkcrC43wOevXVV7nqqqtYsGAB0dHR2Gw2li5dSnx8PHPnzi3uGkXcLmlcg+6NqrNk60Fe+WUz797SzuqSRERERKQSOK8Zp0suuYR///2X66+/niNHjpCcnMyAAQPYuHEjn376aXHXKOJms9kY189cFPfnDQn8sG6f1SWJiIiISCVw3us41apVixdffJHZs2czZ84cXnjhBQ4fPsznn39epPNMmTKFevXq4ePjQ/v27VmyZMlZ9/3zzz/p2rUr1apVw9fXl6ioKCZNmnS+b0HKqWa1ghh5aUMAnpqzgfhkXbInIiIiIiXrvINTcZg5cyZjxozhqaeeYs2aNXTv3p1+/foRFxeX7/7+/v6MHDmSxYsXExsby//93//xf//3f3zwwQelXLlYbXSvRrSvU4W0zBxGz1hDtlNrO4mIiIhIyTmvduRns27dOtq1a4fT6SzU/p06daJdu3ZMnTrVPda0aVP69+/PhAkTCnWOAQMG4O/vzxdffFGo/dWOvOLYcziDfm8uIe14DiMubcCjffN2ehQREREROZsSb0deHLKysli1ahV9+vTJNd6nTx+WLl1aqHOsWbOGpUuXcskll5x1n8zMTFJTU3NtUjHUruLHywPMLo5TFm5n6baDFlckIiIiIhVVkbrqDRgwoMDnjxw5UuhzHTx4EKfTSWhoaK7x0NBQEhMTCzy2du3aHDhwgJycHJ577jnuvvvus+47YcIEnn/++ULXJeXLVa3C+XNbBNP/jmfMzLXMe7A71QK8rS5LRERERCqYIs04BQcHF7jVqVOH2267rUgF2Gy2XI8Nw8gzdqYlS5awcuVK3nvvPSZPnsz06dPPuu+4ceNISUlxb/Hx8UWqT8q+Z65uTsOaASSlZfLot+spxqtPRURERESAIs44FWer8erVq+NwOPLMLiUlJeWZhTpTvXr1AGjZsiX79+/nueee46abbsp3X29vb7y9NQNRkfl6OXj7prZc9+5f/L45ic+W7uKOrvWsLktEREREKhDL7nHy8vKiffv2xMTE5BqPiYmhS5cuhT6PYRhkZmYWd3lSzjQND+L/rmoKwIS5m9m4L8XiikRERESkIinSjFNxGzt2LEOHDqVDhw5ER0fzwQcfEBcXx/DhwwHzMru9e/cybdo0AN59910iIyOJijK7p/3555+8/vrrjBo1yrL3IGXH0M51WPzvQRbE7mfU9DX8NKobfl6W/hEXERERkQrC0u8qBw8ezKFDhxg/fjwJCQm0aNGCuXPnUqdOHQASEhJyrenkcrkYN24cO3fuxMPDgwYNGvDyyy9z3333WfUWpAyx2Wy8dkMr+r25hB0H0nn+h028ckMrq8sSERERkQqgWNdxKg+0jlPFt2z7IW7+aDmGAW/f1JZrWteyuiQRERERKYPKxTpOIiUlukE1Rl7aEIAn52wgPjnD4opEREREpLxTcJIK6cFejWhfpwppmTmMnrGGbKfL6pJEREREpBxTcJIKycNh580hbQj08WBN3BEmL/jX6pJEREREpBxTcJIKq3YVP14eYDaHmLJwO0u3HbS4IhEREREprxScpEK7qlU4N3WMwDBgzMy1JKdnWV2SiIiIiJRDCk5S4T1zdXMa1gwgKS2TR2eto5I1khQRERGRYqDgJBWer5eDt4a0xcvDzm+bk/hs6S6rSxIRERGRckbBSSqFZrWCeOrKpgBMmLuZjftSLK5IRERERMoTBSepNG6LrkPvpqFkOV2Mmr6GjKwcq0sSERERkXJCwUkqDZvNxms3tCIsyIcdB9J5/odNVpckIiIiIuWEgpNUKlX8vZg0uA02G8xcGc+P6/ZZXZKIiIiIlAMKTlLpRDeoxshLGwLw5JwNxCdnWFyRiIiIiJR1Ck5SKT3YqxHtIkNIy8xh9Iw1ZDtdVpckIiIiImWYgpNUSh4OO28OaUugjwdr4o4wecG/VpckIiIiImWYgpNUWhFV/Xh5QCsApizcztJtBy2uSERERETKKgUnqdSuahXOTR0jMAwYM3MtyelZVpckIiIiImWQgpNUes9c3ZyGNQNISsvk0VnrMAzD6pJEREREpIxRcJJKz9fLwVtD2uLlYee3zUl8vnSX1SWJiIiISBmj4CQCNKsVxFNXNgXgpbmb2bgvxeKKRERERKQsUXASOeG26Dr0blqTLKeLUdPXkJGVY3VJIiIiIlJGKDiJnGCz2Xj1htaEBnmz40A6z/+wyeqSRERERKSMUHASOU1Vfy8mD26LzQYzV8bz47p9VpckIiIiImWAgpPIGaIbVGPkpQ0BeHLOBuKTMyyuSERERESspuAkko8HezWiXWQIaZk5jJ6xhmyny+qSRERERMRCCk4i+fBw2HlzSFsCfTxYE3eEyQv+tbokEREREbGQgpPIWURU9ePlAa0AmLJwO0u3HbS4IhERERGxioKTSAGuahXOkIsjMAwYM3MtyelZVpckIiIiIhZQcBI5h2euaUaDGv4kpWXy6Kx1GIZhdUkiIiIiUsoUnETOwc/Lg7dvaoeXh53fNifx+dJdVpckIiIiIqVMwUmkEJrVCuKpK5sC8NLczWzal2pxRSIiIiJSmhScRArptug69G5akyyni1HTV5ORlWN1SSIiIiJSShScRArJZrPx6g2tCQ3yZvuBdJ7/YZPVJYmIiIhIKVFwEimCqv5eTBrcBpsNZq6M58d1+6wuSURERERKgYKTSBF1aVCdET0bAvDknA3EJ2dYXJGIiIiIlDQFJ5Hz8GDvRrSLDCEtM4fRM9aQ7XRZXZKIiIiIlCAFJ5Hz4Omw8+aQtgT6eLAm7ghvLthqdUkiIiIiUoIUnETOU0RVP14e0AqAdxduY+n2gxZXJCIiIiIlRcFJ5AJc1SqcIRdHYBjw0My1JKdnWV2SiIiIiJQABSeRC/TMNc1oUMOf/amZPDprHYZhWF2SiIiIiBQzBSeRC+Tn5cHbN7XDy2Hnt81JfL50l9UliYiIiEgxU3ASKQbNagXx5JVRALw0dzOb9qVaXJGIiIiIFCcFJ5FicnuXuvRuWpMsp4tR01eTkZVjdUkiIiIiUkwUnESKic1m49UbWhMa5M32A+mM/3GT1SWJiIiISDFRcBIpRlX9vZg0uA02G8xYEc9P6/dZXZKIiIiIFAMFJ5Fi1qVBdUb0bAjAuDkbiE/OsLgiEREREblQCk4iJeDB3o1oFxlC2vEcRs9YQ7bTZXVJIiIiInIBLA9OU6ZMoV69evj4+NC+fXuWLFly1n3nzJnD5ZdfTo0aNQgKCiI6Oppff/21FKsVKRxPh503h7Ql0NuDNXFHeHPBVqtLEhEREZELYGlwmjlzJmPGjOGpp55izZo1dO/enX79+hEXF5fv/osXL+byyy9n7ty5rFq1iksvvZRrrrmGNWvWlHLlIucWUdWPCQNbAvDuwm0s3X7Q4opERERE5HzZDMMwrHrxTp060a5dO6ZOneoea9q0Kf3792fChAmFOkfz5s0ZPHgwzzzzTKH2T01NJTg4mJSUFIKCgs6rbpGieGL2emasiCc0yJt5D/agqr+X1SWJiIiICEXLBpbNOGVlZbFq1Sr69OmTa7xPnz4sXbq0UOdwuVykpaVRtWrVs+6TmZlJampqrk2kND1zTTMa1PBnf2omj327Dgt/ViEiIiIi58my4HTw4EGcTiehoaG5xkNDQ0lMTCzUOSZOnEh6ejqDBg066z4TJkwgODjYvUVERFxQ3SJF5eflwds3tcPLYWdBbBKfL91ldUkiIiIiUkSWN4ew2Wy5HhuGkWcsP9OnT+e5555j5syZ1KxZ86z7jRs3jpSUFPcWHx9/wTWLFFWzWkE8eWUUAC/N3cymfZr5FBERESlPLAtO1atXx+Fw5JldSkpKyjMLdaaZM2dy11138c0339C7d+8C9/X29iYoKCjXJmKF27vUpVdUTbKcLkZNX01GVo7VJYmIiIhIIVkWnLy8vGjfvj0xMTG5xmNiYujSpctZj5s+fTrDhg3j66+/5qqrrirpMkWKjc1m47UbW1Mz0JvtB9IZ/+Mmq0sSERERkUKy9FK9sWPH8tFHH/HJJ58QGxvLQw89RFxcHMOHDwfMy+xuu+029/7Tp0/ntttuY+LEiXTu3JnExEQSExNJSUmx6i2IFElVfy8mD2mDzQYzVsTz0/p9VpckIiIiIoVgaXAaPHgwkydPZvz48bRp04bFixczd+5c6tSpA0BCQkKuNZ3ef/99cnJyGDFiBOHh4e7twQcftOotiBRZlwbVGdGzIQDj5mwgPjnD4opERERE5FwsXcfJClrHScqCbKeLwe8vY3XcEdpFhjDzvmg8HZb3ahERERGpVMrFOk4ilZmnw86bQ9oS6O3B6rgjvLlgq9UliYiIiEgBFJxELBJR1Y+XBrQE4N2F21i6/aDFFYmIiIjI2Sg4iVjomta1GNwhAsOAh2auJTk9y+qSRERERCQfCk4iFnv22mbUr+HP/tRMHvt2HZXstkMRERGRckHBScRifl4evH1TW7wcdhbEJjFt2W6rSxIRERGRMyg4iZQBzWsF8+SVUQC8ODeWTftSLa5IRERERE6n4CRSRtzepS69omqSleNi1PTVZGTlWF2SiIiIiJyg4CRSRthsNl67sTU1A73ZfiCd8T9usrokERERETlBwUmkDKnq78XkwW2w2WDGinh+Wr/P6pJEREREBAUnkTKnS8PqPNCzAQDj5mwgPjnD4opERERERMFJpAwa07sxbSNDSDuew4Mz1pDjdFldkoiIiEilpuAkUgZ5Ouy8NaQtgd4erI47wpu/bbW6JBEREZFKTcFJpIyKqOrHSwNaAvDOH9tYuv2gxRWJiIiIVF4KTiJl2DWtazG4QwSGAQ/NXEtyepbVJYmIiIhUSgpOImXcs9c2o34Nf/anZvLYt+swDMPqkkREREQqHQUnkTLOz8uDt29qi5fDzoLYJKYt2211SSIiIiKVjoKTSDnQvFYw466MAuDFubFs2pdqcUUiIiIilYuCk0g5MaxLXXpF1SQrx8Wo6avJyMqxuiQRERGRSkPBSaScsNlsvHZja2oGerP9QDr/+WmT1SWJiIiIVBoKTiLlSFV/LyYPboPNBtP/jufn9QlWlyQiIiJSKSg4iZQzXRpW54GeDQB4Ys564pMzLK5IREREpOJTcBIph8b0bkzbyBDSjufw4Iw15DhdVpckIiIiUqEpOImUQ54OO28NaUugtwer447w5m9brS5JREREpEJTcBIppyKq+vHSgJYAvPPHNpZtP2RxRSIiIiIVl4KTSDl2TetaDOpQG8OAMTPXkJyeZXVJIiIiIhWSgpNIOffctc2pX8Of/amZPPbtOgzDsLokERERkQpHwUmknPPz8uDtm9ri5bCzIDaJact2W12SiIiISIWj4CRSATSvFcy4K6MAeHFuLJv2pVpckYiIiEjFouAkUkEM61KXy6JqkpXjYtT01WRk5VhdkoiIiEiFoeAkUkHYbDZeu6EVNQO92X4gnf/8tMnqkkREREQqDAUnkQqkWoA3kwa3wWaD6X/H8/P6BKtLEhEREakQFJxEKpiuDatz/yUNAHhiznr2HM6wuCIRERGR8k/BSaQCeujyxrSNDCHteA4PzlhLjtNldUkiIiIi5ZqCk0gF5Omw89aQtgR6e7Bq92He/G2r1SWJiIiIlGsKTiIVVERVP14a0BKAd/7YxrLthyyuSERERKT8UnASqcCuaV2LQR1qYxgwZuYaktOzrC5JREREpFxScBKp4J67tjn1a/izPzWTx75dj2EYVpckIiIiUu4oOIlUcH5eHrw1pC1eDjsLYvfzxfLdVpckIiIiUu4oOIlUAi0uCuaJflEAvPBTLNOW7cLl0syTiIiISGEpOIlUEnd0rctVLcPJcrp45r8bufXj/xGfrDWeRERERApDwUmkkrDZbLx9U1uev7Y5vp4Olm4/xBWTFzP97zjd9yQiIiJyDgpOIpWI3W7j9i51mfdgdzrUqUJ6lpNxczZw+6crSEg5ZnV5IiIiImWWgpNIJVS3uj8z74vm/65qireHncX/HqDPpMXMWhmv2ScRERGRfCg4iVRSDruNu7vX5+fR3WkTEULa8Rwe/XY9d3++kqTU41aXJyIiIlKmKDiJVHINawbw7fBoHruiCV4OO79tTuLySYv579q9mn0SEREROUHBSUTwcNh5oGdDfhzVjRYXBZFyLJsHZ6zl/i9Xc/BoptXliYiIiFjO8uA0ZcoU6tWrh4+PD+3bt2fJkiVn3TchIYGbb76ZJk2aYLfbGTNmTOkVKlIJNAkL5LsHujL28sZ42G38sjGRPpMWM3dDgtWliYiIiFjK0uA0c+ZMxowZw1NPPcWaNWvo3r07/fr1Iy4uLt/9MzMzqVGjBk899RStW7cu5WpFKgdPh53RvRrx35FdiQoLJDk9iwe+Ws3Ir1dzOD3L6vJERERELGEzLLyJoVOnTrRr146pU6e6x5o2bUr//v2ZMGFCgcf27NmTNm3aMHny5CK9ZmpqKsHBwaSkpBAUFHQ+ZYtUGlk5Lt7+fStTFm7H6TKoHuDNS9e3oE/zMKtLExEREblgRckGls04ZWVlsWrVKvr06ZNrvE+fPixdurTYXiczM5PU1NRcm4gUjpeHnYf7NGHO/V1oVDOAg0czufeLVYyduZaUjGyryxMREREpNZYFp4MHD+J0OgkNDc01HhoaSmJiYrG9zoQJEwgODnZvERERxXZukcqidUQIP47qxvBLGmC3wZw1e+kzeRF/bEmyujQRERGRUmF5cwibzZbrsWEYecYuxLhx40hJSXFv8fHxxXZukcrEx9PBE/2imDW8C/Wr+7M/NZM7Pl3B49+uJ/W4Zp9ERESkYrMsOFWvXh2Hw5FndikpKSnPLNSF8Pb2JigoKNcmIuevfZ0q/Dy6O3d2rYfNBjNXxnPFpMX8ufWg1aWJiIiIlBjLgpOXlxft27cnJiYm13hMTAxdunSxqCoRKQxfLwfPXNOMGfd0JrKqH/tSjnPrx//j/77fQHpmjtXliYiIiBQ7Sy/VGzt2LB999BGffPIJsbGxPPTQQ8TFxTF8+HDAvMzutttuy3XM2rVrWbt2LUePHuXAgQOsXbuWTZs2WVG+SKXXqX41fhnTndui6wDw5fI4rnhzMct3HLK4MhEREZHiZWk7cjAXwH311VdJSEigRYsWTJo0iR49egAwbNgwdu3axcKFC93753f/U506ddi1a1ehXk/tyEVKxl/bDvLYt+vZe+QYAMO61OXxK6Lw9XJYXJmIiIhI/oqSDSwPTqVNwUmk5KQdz+alubFM/9tswlK3mh+v39iaDnWrWlyZiIiISF7lYh0nEal4An08mTCgFZ/f2ZGwIB92HcrgxveX8dLcWI5nO60uT0REROS8KTiJSLG7pHENfn2oBze0r41hwAeLd3DVW0tYG3/E6tJEREREzouCk4iUiGBfT16/sTUf3daBGoHebD+QzoApf/Har5vJzNHsk4iIiJQvCk4iUqJ6Nwtl/pgeXNemFi4D3v1jO9e+/Rf/7E2xujQRERGRQlNwEpESV8XfizeHtGXqLe2o5u/Flv1p9H/3LybF/Eu202V1eSIiIiLnpOAkIqWmX8tw5j/UgytbhpHjMnjzt630f/cvNiemWl2aiIiISIEUnESkVFUL8Obdm9vx9k1tCfHzZOO+VK55+0/e/WMbOZp9EhERkTJKwUlESp3NZuOa1rWY/1APejcNJdtp8NqvWxg4dSnbktKsLk9EREQkDwUnEbFMzUAfPrytPW8Mak2gjwfr9qRw5Vt/8v6i7ThdlWptbhERESnjFJxExFI2m40B7WoT89Al9GxSg6wcFxPmbWbQ+8vYeTDd6vJEREREAAUnESkjwoJ9+HTYxbwysCUB3h6s2n2Yfm8u5pM/d+LS7JOIiIhYTMFJRMoMm83G4Isj+fWhHnRrWJ3j2S7G/7SJIR8uJ+5QhtXliYiISCWm4CQiZc5FIb58cVdH/tO/BX5eDv7emcwVby7mi+W7NfskIiIillBwEpEyyWazMbRzHX55sAed6lUlI8vJ09//w9BP/sfeI8esLk9EREQqGQUnESnTIqv5Mf2ezjx7TTN8PO38te0QfSctZuaKOAxDs08iIiJSOhScRKTMs9tt3NG1HnNHd6ddZAhHM3N4fPYG7vhsBYkpx60uT0RERCoBBScRKTfq1whg1vAuPHllFF4edhZuOcDlkxYxe9UezT6JiIhIiVJwEpFyxWG3cW+PBvw8qhutaweTdjyHh2et455pq0hK0+yTiIiIlAwFJxEplxqFBjL7/i482rcJng4bC2L302fSYn5Yt0+zTyIiIlLsFJxEpNzycNgZcWlDfhzVjea1gjiSkc3o6WsY8fVqDh3NtLo8ERERqUAUnESk3IsKC+L7EV15sFcjPOw25m5IpM+kxfzyT4LVpYmIiEgFoeAkIhWCp8POQ5c35vsRXWkSGsih9CyGf7ma0dPXcDg9y+ryREREpJxTcBKRCqXFRcH8MKorIy5tgN0GP6zbR5/Ji1mwab/VpYmIiEg5puAkIhWOt4eDR/tGMeeBrjSo4c+BtEzunraSh79ZR8qxbKvLExERkXJIwUlEKqw2ESH8PLo79/aoj80Gs1fvoe+kxSzckmR1aSIiIlLOKDiJSIXm4+ngySubMuu+aOpW8yMx9TjDPl3BuDnrSTuu2ScREREpHAUnEakUOtStyrwHezCsS10Apv8dzxWTl/DXtoPWFiYiIiLlgoKTiFQavl4Onru2OTPu7UxEVV/2HjnGLR/9j6e//4f0zByryxMREZEyTMFJRCqdzvWr8cuDPbilUyQAXyzfTb83l/D3zmSLKxMREZGySsFJRColf28PXry+JV/c1ZFawT7EJWcw+INljP9xE8eynFaXJyIiImWMgpOIVGrdG9Xgl4d6MLhDBIYBn/y1k6veWsKq3YetLk1ERETKEAUnEan0gnw8eeWGVnx6x8WEBnmz42A6N763lAnzYjmerdknERERUXASEXG7tElN5o+5hAFtL8JlwPuLdnDN23+yfs8Rq0sTERERiyk4iYicJtjPkzcGt+GDoe2pHuDF1qSjXD9lKa//uoW9R45hGIbVJYqIiIgFbEYl+y4gNTWV4OBgUlJSCAoKsrocESnDktOzePaHjfy4bp97LDTIm7YRVWgbGULbyCq0vCgYXy+HhVWKiIjI+SpKNlBwEhE5h7kbEpi6cDubElJxunL/k+mw22gaHpgrTNWt5ofNZrOoWhERESksBacCKDiJyPk6luVkw94U1sQdZk3cEVbHHSYpLTPPfiF+nrSNMENU28gQWkeEEOTjaUHFIiIiUhAFpwIoOIlIcTEMg4SU4+4QtSbuMP/sSyUrx5VrP5sNGtYIcM9ItY0MoVHNQBx2zUqJiIhYScGpAApOIlKSMnOcxCakuWel1sQfJj75WJ79/L0ctI4IMcNURBXaRIZQPcDbgopFREQqLwWnAig4iUhpO5CWydr4I+4wtW7PETKy8q4PFVnV70SQMmemmoYH4eWh5qciIiIlRcGpAApOImI1p8vg3/1p5oxU3GHWxB9hW9LRPPt5edhpeVFwrvulwoN91HhCRESkmCg4FUDBSUTKopRj2ayLP+K+vG9N3BFSjmXn2U/t0EVERIqPglMBFJxEpDwwDIOdB9NzBanNiWl52qF72G00DQ86EaTM+6XqqB26iIhIoSg4FUDBSUTKq4ysHDbsSWHNifulVscd4UA+7dCr+Hmal/aduMSvVUSw2qGLiIjkQ8GpAApOIlJRGIbBvpTjpzr4xR3mn72pZDnztkNvVDMg1yV+DWsGqB26iIhUegpOBVBwEpGKrLDt0AO8PWgdEewOU20iQqimdugiIlLJlKvgNGXKFF577TUSEhJo3rw5kydPpnv37mfdf9GiRYwdO5aNGzdSq1YtHnvsMYYPH17o11NwEpHKprDt0OtU88vVwS8qTO3QRUSkYis3wWnmzJkMHTqUKVOm0LVrV95//30++ugjNm3aRGRkZJ79d+7cSYsWLbjnnnu47777+Ouvv3jggQeYPn06AwcOLNRrKjiJSGVX2Hbo3ifboUee3g7d14KKRURESka5CU6dOnWiXbt2TJ061T3WtGlT+vfvz4QJE/Ls//jjj/PDDz8QGxvrHhs+fDjr1q1j2bJlhXpNBScRkbwK2w49LMjnVAe/E+3QfTzVDl1ERMqnomQDj1KqKY+srCxWrVrFE088kWu8T58+LF26NN9jli1bRp8+fXKN9e3bl48//pjs7Gw8PdU1SkTkfAT7etKjcQ16NK4BnL0demLqceb9k8i8fxIBtUMXEZHKw7LgdPDgQZxOJ6GhobnGQ0NDSUxMzPeYxMTEfPfPycnh4MGDhIeH5zkmMzOTzMxT7XpTU1OLoXoRkYrNZrNRv0YA9WsEMLB9beDs7dA37E1hw94Upi3bDUBVfy/aRIS475dqcVEQIX5eVr4dERGRC2ZZcDrpzJ9KGoZR4E8q89s/v/GTJkyYwPPPP3+BVYqIiJ+XB53qV6NT/WrA2duhJ6dn8fvmJH7fnOQ+NizIh6jwQJqEBdI0LIgmYYE0qBGg5hMiIlJuWBacqlevjsPhyDO7lJSUlGdW6aSwsLB89/fw8KBatWr5HjNu3DjGjh3rfpyamkpERMQFVi8iIjabjYtCfLkoxJerW9UCzt4OPTH1OImpx1m45YD7eA+7jQY1AogKDyQqLIiosECiwgMJC/LRpX4iIlLmWBacvLy8aN++PTExMVx//fXu8ZiYGK677rp8j4mOjubHH3/MNTZ//nw6dOhw1vubvL298fbW2iQiIqXB28NBmwhzXag7uppjacez+Xd/GrEJaWxJTGNzYiqbE9NIO57Dlv1pbNmfxn/Z5z5HkI8HUeEngtSJ2akmYYEEeFt+kYSIiFRiZaId+XvvvUd0dDQffPABH374IRs3bqROnTqMGzeOvXv3Mm3aNOBUO/L77ruPe+65h2XLljF8+HC1IxcRKWdOXua3JTE1V6DafiAdpyv//5Yiqvqempk6EajqVffHYdfslIiInJ9y0VUPYPDgwRw6dIjx48eTkJBAixYtmDt3LnXq1AEgISGBuLg49/716tVj7ty5PPTQQ7z77rvUqlWLt956q9ChSUREyobTL/O7LOrU5dmZOU62J6WzOTGVLYlpxCamsSUxlf2pmcQnHyM++Rgxm/a79/f2sNM41JyROhmoosIDqR6gKw1ERKR4WTrjZAXNOImIlD+H07PYfGJW6mSg+jcxjWPZznz3rx7g5Z6VOhmoGoUGaM0pERHJpdwsgGsFBScRkYrB5TKIS87IFag2J6ax61A6+f3PZrdB3er+7q5+UWGBNA0P4qIQX+y63E9EpFJScCqAgpOISMWWkZXD1v1HT8xMpbI5wQxWhzOy893f38txogFFEE3DA2kSas5QBftpUXURkYpOwakACk4iIpWPYRgcSMt0z05tTkxjc0Ia25KOkuV05XtMeLAPUacHqrBA6lfX2lMiIhWJglMBFJxEROSkbKeLXQfTc98/lZDG3iPH8t3f03Fi7amwQKLCg9wL+oYGeWvtKRGRckjBqQAKTiIici6px7P598Q9U+77pxLSSMvMyXf/YF/PE00oTgWqJqGB+GvtKRGRMk3BqQAKTiIicj4Mw2DvkWPuJhTm5X6p7Dh49rWnIqv65QlUdatp7SkRkbJCwakACk4iIlKcMnOcbEs6midQJaVl5ru/j6edRjVPhamTwaqa1p4SESl1Ck4FUHASEZHSkJyelesyv837z7X2lPeprn7hQdSt5keInxchfp6E+Hri4VBTChGR4qbgVAAFJxERsYrzxNpTWxJTiU1IOzFLlcru5Ix81546XaCPByF+nlTx8yLY1/w1xM/TDFe+nlTx9yTE18u9T4ifJ0E+nlqjSkSkAApOBVBwEhGRsiYjK4d/9x91B6rNiansO3KcIxlZpB7PvyFFYdhsuEOW+avnabNYXlTx93Q/fyqIeRLg7aEugSJSKRQlG6jdj4iIiMX8vDxoExFCm4iQPM/lOF2kHs/hcEYWRzKyOXLiV/fjY1kczsgm5fSxjCzSs5wYBice57/479l42G2E+HmeZWbrtODl50nwaTNcvp4OBS4RqbAUnERERMowD4edqv5eVPX3KtJxWTkujhzLOhGozFDlDlfH8glgJ0LY8WwXOS6Dg0ezOHg0C0gv9Gt6edjNcOXndSJQnbh80P9EuPI9NeN1+gyXt4ejiL8rIiKlT8FJRESkAvLysFMz0IeagT5FOu54ttMdqE6FLTNUnZzNyjXDdSKEZTsNsnJcJKVlnrWj4Nn4ejpOzF6Zs1h5gteJGa8qJ4JWyIlLDz3VMENKgGEYZDsNsp0usnJc5q9OV66xLKeL7BxzzMNhI8DbA39vD/y9Hfh7eeDnpdnXikjBSURERNx8PB2EBTsICy584DIMg4wsZ67ZK3eoSjd/zTvjZQYulwHHsp0cS3GyL+V4kWoN9PbIdalgiJ8XQT4eeDrsOOw2POy203614+GwYbedNu444/lc+598Pve4Pdd57fnsn8/4iV/1jbTZIMUdRE4ED3cQOW3LyjFO2+dUcDkZZE6d49T5Tn8u3/OeOGee/fIZu1A2G/h7nQhS3h5msPLyOPH1aWPep8b8vDxyBbAA93MeeHvY9eenDFBwEhERkQtis9nc3wDWrlL441wug7TMnFyXDaYcy+Zweu5wZc54nfg6/VTDjLTMHNIyc9hz+FgJvbPi5bDbcNhOC1OOs4ewswU3x5nhrgjh7VTos+fZ32GzkXMi1JyaVTnx+LTwkpVj5A44ToOsHGf+MzSnhZqTgeRsi0WXZTYbeDnseDnseHqc/NWGp8OOp91OtstFemYO6ZlO0rNyMAwwDDiamcPRzBygaDOw+XHYbfh7Oc4IW6dmuE4PYLmfP23stGDm5aHZ2vOh4CQiIiKWsNttBPuaTSjqVCv8cU6XQcqx0y4bPJbF4XQzXKUdz8bpMshxmd+k5zgNnC7Xqce5fnWdeP70cRcuF+S4XPnsbz7vdJ5l3GUUOFvhdBk4MSD/pbwqJQ+7GUC8POzmrw4bnu6vTwaVEyHlxObtYcfz5NiJION1+tjJYx02vDwcJ349dfzJfb0cucdyBSLHqfM6itDS3+UyOJbtJP1EaDoZpnI9dn+dQ3pWDkfPHMs0xzKycsjIMv+wOF0GqcdzLqjL5um8HHb8ToQudwDLFbbyny3Ls9+JWbXKss6cgpOIiIiUKw677bwaZpQW14lA5TJOBCuncZYgZga6/MKb+7HTwGmcK+ydFg7zhLpTodHpKjg8epwIHl65AoYZJs4cyxVm3OElv4BzKtCcCicnxuz2CrfOmN1+ava1ZjGcz+kyyMgyA1fuYJU7dJ0cy8h0cjQrdwA7/fnMHBeAOXuY4Spyx82z8faw57n00P+0cOXnfeZsmRnaLq5blSpl9O9xfhScRERERIqR3W7Dq4IFArGGw24j0MeTQB/PYjlfttOVK1ydCmPO02bATo0dzcy738nQlp6Z455hzcxxkZmTxaH0rCLV8+3waDr4Vy2W91YaFJxERERERCoBT4edYD87wX7FE8Sycly5ZsBOn+U689LDXDNjWWYAK0+zTaDgJCIiIiIi58HLw46Xh1e5C0Dnq3LcySUiIiIiInIBFJxERERERETOQcFJRERERETkHBScREREREREzkHBSURERERE5BwUnERERERERM5BwUlEREREROQcFJxERERERETOQcFJRERERETkHBScREREREREzkHBSURERERE5BwUnERERERERM5BwUlEREREROQcFJxERERERETOQcFJRERERETkHBScREREREREzkHBSURERERE5BwUnERERERERM7Bw+oCSpthGACkpqZaXImIiIiIiFjpZCY4mREKUumCU1paGgAREREWVyIiIiIiImVBWloawcHBBe5jMwoTryoQl8vFvn37CAwMxGazWV1OuZWamkpERATx8fEEBQVZXY6UMH3elYs+78pFn3flo8+8ctHnXTDDMEhLS6NWrVrY7QXfxVTpZpzsdju1a9e2uowKIygoSH8JKxF93pWLPu/KRZ935aPPvHLR531255ppOknNIURERERERM5BwUlEREREROQcFJzkvHh7e/Pss8/i7e1tdSlSCvR5Vy76vCsXfd6Vjz7zykWfd/GpdM0hREREREREikozTiIiIiIiIueg4CQiIiIiInIOCk4iIiIiIiLnoOAkIiIi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X5YtWwaAzWbj+++/t7ZIERGpUDysLkBERKSoBg4cSHZ2Np9//jn169dn//79/PbbbyQnJ1tdmoiIVFCacRIRkXLlyJEj/Pnnn7zyyitceuml1KlTh44dOzJu3Diuuuoq6tatC8D111+PzWZzPwb48ccfad++PT4+PtSvX5/nn3+enJwc9/M2m42pU6fSr18/fH19qVevHrNmzXI/n5WVxciRIwkPD8fHx4e6desyYcKE0nrrIiJiIQUnEREpVwICAggICOD7778nMzMzz/MrVqwA4NNPPyUhIcH9+Ndff+XWW29l9OjRbNq0iffff5/PPvuMF198MdfxTz/9NAMHDmTdunXceuut3HTTTcTGxgLw1ltv8cMPP/DNN9+wZcsWvvzyy1zBTEREKi6bYRiG1UWIiIgUxezZs7nnnns4duwY7dq145JLLmHIkCG0atUKMGeOvvvuO/r37+8+pkePHvTr149x48a5x7788ksee+wx9u3b5z5u+PDhTJ061b1P586dadeuHVOmTGH06NFs3LiRBQsWYLPZSufNiohImaAZJxERKXcGDhzIvn37+OGHH+jbty8LFy6kXbt2fPbZZ2c9ZtWqVYwfP949YxUQEMA999xDQkICGRkZ7v2io6NzHRcdHe2ecRo2bBhr166lSZMmjB49mvnz55fI+xMRkbJHwUlERMolHx8fLr/8cp555hmWLl3KsGHDePbZZ8+6v8vl4vnnn2ft2rXubcOGDWzduhUfH58CX+vk7FK7du3YuXMn//nPfzh27BiDBg3ihhtuKNb3JSIiZZOCk4iIVAjNmjUjPT0dAE9PT5xOZ67n27Vrx5YtW2jYsGGezW4/9d/h8uXLcx23fPlyoqKi3I+DgoIYPHgwH374ITNnzmT27Nnq5iciUgmoHbmIiJQrhw4d4sYbb+TOO++kVatWBAYGsnLlSl599VWuu+46AOrWrctvv/1G165d8fb2pkqVKjzzzDNcffXVREREcOONN2K321m/fj0bNmzghRdecJ9/1qxZdOjQgW7duvHVV1/x999/8/HHHwMwadIkwsPDadOmDXa7nVmzZhEWFkZISIgVvxUiIlKKFJxERKRcCQgIoFOnTkyaNInt27eTnZ1NREQE99xzD08++SQAEydOZOzYsXz44YdcdNFF7Nq1i759+/LTTz8xfvx4Xn31VTw9PYmKiuLuu+/Odf7nn3+eGTNm8MADDxAWFsZXX31Fs2bN3K/9yiuvsHXrVhwOBxdffDFz587NNWMlIiIVk7rqiYiInJBfNz4RERHQPU4iIiIiIiLnpOAkIiIiIiJyDrrHSURE5ARdvS4iImejGScREREREZFzUHASERERERE5BwUnERERERGRc1BwEhEREREROQcFJxERERERkXNQcBIRERERETkHBScREREREZFzUHASERERERE5BwUnERERERGRc/h/gsMJQ5ZW90cAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "class CausalSft_Model_No(nn.Module):\n",
    "    def __init__(self, base_model, num_labels, max_words):\n",
    "        super().__init__()\n",
    "\n",
    "        self.max_words = max_words\n",
    "        self.base_model = copy.deepcopy(base_model)\n",
    "        self.sft_model = copy.deepcopy(base_model)\n",
    "        self.sft_classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "        self.r_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 2, self.base_model.config.hidden_size // 2),\n",
    "        )\n",
    "        self.c_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "        )\n",
    "        self.num_patches = 10\n",
    "        self.patches_size = self.max_words // self.num_patches\n",
    "        self.patch_extractor = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches, self.base_model.config.hidden_size * self.num_patches // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches // 2, self.base_model.config.hidden_size),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size),\n",
    "        )\n",
    "        self.classifier = nn.Linear(self.base_model.config.hidden_size + self.base_model.config.hidden_size // 4, num_labels)\n",
    "\n",
    "        self.cross_entropy_loss = nn.CrossEntropyLoss()\n",
    "        self.mse_loss = nn.MSELoss()\n",
    "\n",
    "        for param in self.base_model.parameters():\n",
    "            param.requires_grad = False\n",
    "\n",
    "        print(f'Initialise Causal SFT model \"{model_name}\" (unfreezed all layers) with a linear head!')\n",
    "        count_parameters(self)\n",
    "\n",
    "    def entropy_maximization(self, feature):\n",
    "        p = F.softmax(feature, dim=-1)\n",
    "        log_p = F.log_softmax(feature, dim=-1)\n",
    "        entropy = -torch.mean(torch.mean(p * log_p, dim=-1))  # Maximize entropy\n",
    "        return -entropy  # Minimize negative entropy\n",
    "\n",
    "    def forward(self, input_ids1, attention_mask1, input_ids2, attention_mask2, input_ids3, attention_mask3, input_ids4, attention_mask4, labels=None):\n",
    "        B, _ = input_ids1.size()\n",
    "        # avoid unwanted computational graph\n",
    "        outputs_sft2 = self.sft_model(input_ids2, attention_mask=attention_mask2)[1]\n",
    "        outputs_sft2_logits = self.sft_classifier(outputs_sft2)\n",
    "        # pull out R0 and R1 using another data point\n",
    "        outputs_base3 = self.base_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        outputs_sft3 = self.sft_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        # optimise C from R0 and R1\n",
    "        base_C3= self.r_2_c(outputs_base3)[:, :self.base_model.config.hidden_size//4]\n",
    "        sft_C3 = self.r_2_c(outputs_sft3)[:, :self.base_model.config.hidden_size//4]\n",
    "        # gather C \n",
    "        outputs_sft = self.sft_model(input_ids1, attention_mask=attention_mask1)[1]\n",
    "        sft_C = self.r_2_c(outputs_sft)[:, :self.base_model.config.hidden_size//4]\n",
    "        # extract non-contextual embeddings\n",
    "        with torch.no_grad():\n",
    "            embeddings_1 = self.sft_model.embeddings(input_ids1, attention_mask1)[:, :self.max_words]\n",
    "            embeddings_2 = self.sft_model.embeddings(input_ids2, attention_mask2)[:, :self.max_words]\n",
    "            embeddings_4 = self.sft_model.embeddings(input_ids4, attention_mask4)[:, :self.max_words]\n",
    "\n",
    "        input = embeddings_1\n",
    "        patches = torch.sum(input.view(B, self.num_patches, self.patches_size, -1), dim=2)\n",
    "        Ai_samples = self.patch_extractor(patches.view(B, -1))\n",
    "        # Ai_samples = Ai_samples[torch.randperm(B).to(input_ids1.device), :]\n",
    "        Ai_Z1 = torch.cat([Ai_samples, self.c_2_c(sft_C)], dim=-1)\n",
    "        logits = self.classifier(Ai_Z1)  \n",
    "\n",
    "        loss = None\n",
    "        if labels is not None:\n",
    "            loss_sft = self.cross_entropy_loss(outputs_sft2_logits, labels) \n",
    "            loss_identify = self.mse_loss(base_C3, sft_C3) + self.entropy_maximization(sft_C3) + self.entropy_maximization(base_C3)\n",
    "            loss_cls = self.cross_entropy_loss(logits, labels) \n",
    "            \n",
    "            loss = loss_sft + loss_identify + loss_cls \n",
    "        \n",
    "        return (loss, logits) if loss is not None else logits\n",
    "    \n",
    "set_seed(data_seed)\n",
    "model = CausalSft_Model_No(base_model, num_cls, max_words)\n",
    "\n",
    "# Define training arguments and trainer\n",
    "training_args = TrainingArguments(\n",
    "    overwrite_output_dir=True,\n",
    "    output_dir=f'./results/yelp/causal-sft-no/{N}-shot-{data_seed}',\n",
    "    eval_strategy=\"steps\",\n",
    "    save_strategy=\"steps\",\n",
    "    logging_strategy=\"steps\",\n",
    "    logging_steps=0.1,\n",
    "    save_steps=0.1,\n",
    "    learning_rate=5e-5,\n",
    "    per_device_train_batch_size=128,\n",
    "    per_device_eval_batch_size=128,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=True,\n",
    "    metric_for_best_model=\"eval_loss\",  # Choose model based on validation loss\n",
    "    greater_is_better=False,  # Lower validation loss is better\n",
    "    save_total_limit=1,\n",
    ")\n",
    "\n",
    "\n",
    "trainer = Trainer(\n",
    "    model=model,\n",
    "    args=training_args,\n",
    "    train_dataset=paired_train_dataset,\n",
    "    eval_dataset=paired_validation_dataset,\n",
    "    tokenizer=tokenizer,\n",
    "    data_collator=paired_data_collator,\n",
    "    compute_metrics=compute_metrics,\n",
    ")\n",
    "\n",
    "# Train the model\n",
    "trainer.train()\n",
    "\n",
    "results = {}\n",
    "train_results = trainer.evaluate(paired_train_dataset)\n",
    "print(f'train: {train_results}')\n",
    "results['train'] = train_results\n",
    "\n",
    "valid_results = trainer.evaluate(paired_validation_dataset)\n",
    "print(f'validation: {valid_results}')\n",
    "results['valid'] = valid_results\n",
    "\n",
    "# Evaluate on the test set\n",
    "test_results_ID_90 = trainer.evaluate(paired_test_dataset_ID_90)\n",
    "print(f'ID 90: {test_results_ID_90}')\n",
    "results['ID 90'] = test_results_ID_90\n",
    "\n",
    "test_results_OOD_change_70 = trainer.evaluate(paired_test_dataset_OOD_change_70)\n",
    "print(f'OOD change 70: {test_results_OOD_change_70}')\n",
    "results['OOD change 70'] = test_results_OOD_change_70\n",
    "\n",
    "test_results_OOD_balanced_50 = trainer.evaluate(paired_test_dataset_OOD_balanced_50)\n",
    "print(f'OOD balanced 50: {test_results_OOD_balanced_50}')\n",
    "results['OOD balanced 50'] = test_results_OOD_balanced_50\n",
    "\n",
    "test_results_OOD_change_30 = trainer.evaluate(paired_test_dataset_OOD_change_30)\n",
    "print(f'OOD change 30: {test_results_OOD_change_30}')\n",
    "results['OOD change 30'] = test_results_OOD_change_30\n",
    "\n",
    "test_results_OOD_flip_10 = trainer.evaluate(paired_test_dataset_OOD_flip_10)\n",
    "print(f'OOD flip: {test_results_OOD_flip_10}')\n",
    "results['OOD flip'] = test_results_OOD_flip_10\n",
    "\n",
    "test_results_OOD_original_0 = trainer.evaluate(paired_test_dataset_OOD_original_0)\n",
    "print(f'OOD original: {test_results_OOD_original_0}')\n",
    "results['OOD original'] = test_results_OOD_original_0\n",
    "\n",
    "# Manually save the results\n",
    "with open(f'./results/yelp/causal-sft-no/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "plot_training_metrics(trainer)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 5. Causal SFT - C"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Initialise Causal SFT model \"bert-base-uncased\" (unfreezed all layers) with a linear head!\n",
      "Total Parameters: 252519172\n",
      "Trainable Parameters: 143036932\n",
      "Percentage of Trainable Parameters: 56.6440%\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='630' max='630' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [630/630 04:30, Epoch 10/10]\n",
       "    </div>\n",
       "    <table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       " <tr style=\"text-align: left;\">\n",
       "      <th>Step</th>\n",
       "      <th>Training Loss</th>\n",
       "      <th>Validation Loss</th>\n",
       "      <th>F1</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <td>63</td>\n",
       "      <td>0.681200</td>\n",
       "      <td>0.475675</td>\n",
       "      <td>0.932989</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>126</td>\n",
       "      <td>0.267400</td>\n",
       "      <td>0.450324</td>\n",
       "      <td>0.930465</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>189</td>\n",
       "      <td>0.108000</td>\n",
       "      <td>0.451854</td>\n",
       "      <td>0.928494</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>252</td>\n",
       "      <td>0.037200</td>\n",
       "      <td>0.539545</td>\n",
       "      <td>0.938499</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>315</td>\n",
       "      <td>0.012700</td>\n",
       "      <td>0.564392</td>\n",
       "      <td>0.934491</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>378</td>\n",
       "      <td>-0.004800</td>\n",
       "      <td>0.548477</td>\n",
       "      <td>0.937495</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>441</td>\n",
       "      <td>-0.017500</td>\n",
       "      <td>0.704018</td>\n",
       "      <td>0.938499</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>504</td>\n",
       "      <td>-0.024200</td>\n",
       "      <td>0.720526</td>\n",
       "      <td>0.933998</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>567</td>\n",
       "      <td>-0.027900</td>\n",
       "      <td>0.720717</td>\n",
       "      <td>0.934995</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>630</td>\n",
       "      <td>-0.031500</td>\n",
       "      <td>0.604161</td>\n",
       "      <td>0.935500</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table><p>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='271' max='63' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [63/63 00:46]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train: {'eval_loss': 0.13284417986869812, 'eval_f1': 0.986249545375594, 'eval_runtime': 10.5664, 'eval_samples_per_second': 757.12, 'eval_steps_per_second': 5.962, 'epoch': 10.0}\n",
      "validation: {'eval_loss': 0.44650939106941223, 'eval_f1': 0.9304647978038882, 'eval_runtime': 3.1091, 'eval_samples_per_second': 643.265, 'eval_steps_per_second': 5.146, 'epoch': 10.0}\n",
      "ID 90: {'eval_loss': 0.4529305100440979, 'eval_f1': 0.9271855546649757, 'eval_runtime': 5.4977, 'eval_samples_per_second': 727.582, 'eval_steps_per_second': 5.821, 'epoch': 10.0}\n",
      "OOD change 70: {'eval_loss': 0.9313404560089111, 'eval_f1': 0.8443569916762946, 'eval_runtime': 5.5179, 'eval_samples_per_second': 724.912, 'eval_steps_per_second': 5.799, 'epoch': 10.0}\n",
      "OOD balanced 50: {'eval_loss': 1.4047414064407349, 'eval_f1': 0.763030063958638, 'eval_runtime': 5.5327, 'eval_samples_per_second': 722.979, 'eval_steps_per_second': 5.784, 'epoch': 10.0}\n",
      "OOD change 30: {'eval_loss': 1.8949098587036133, 'eval_f1': 0.6819235810061499, 'eval_runtime': 5.489, 'eval_samples_per_second': 728.735, 'eval_steps_per_second': 5.83, 'epoch': 10.0}\n",
      "OOD flip: {'eval_loss': 2.408996820449829, 'eval_f1': 0.593997644983846, 'eval_runtime': 5.4942, 'eval_samples_per_second': 728.037, 'eval_steps_per_second': 5.824, 'epoch': 10.0}\n",
      "OOD original: {'eval_loss': 0.8059301376342773, 'eval_f1': 0.8498590851813124, 'eval_runtime': 5.4851, 'eval_samples_per_second': 729.25, 'eval_steps_per_second': 5.834, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA04AAAINCAYAAAAJGy/3AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAAB4jUlEQVR4nO3dd3hUZfrG8Xtm0jsQUoDQe4dQBMRGE0FlwRULIIooKrrIWhbZte26uK4FXRfsoKKIivpTQDEoTdGlSAfpEEpCCCUJ6Zmc3x8nGRJSgSQnmXw/13WuzJw5M/MMJ5rced/zvDbDMAwBAAAAAEpkt7oAAAAAAKjuCE4AAAAAUAaCEwAAAACUgeAEAAAAAGUgOAEAAABAGQhOAAAAAFAGghMAAAAAlIHgBAAAAABl8LC6gKqWm5urY8eOKTAwUDabzepyAAAAAFjEMAylpKSoQYMGsttLH1OqdcHp2LFjioqKsroMAAAAANXE4cOH1ahRo1KPqXXBKTAwUJL5jxMUFGRxNQAAAACskpycrKioKFdGKE2tC0750/OCgoIITgAAAADKdQkPzSEAAAAAoAwEJwAAAAAoA8EJAAAAAMpAcAIAAACAMhCcAAAAAKAMBCcAAAAAKAPBCQAAAADKQHACAAAAgDIQnAAAAACgDAQnAAAAACgDwQkAAAAAykBwAgAAAIAyEJwAAAAAoAwEJwAAAAAoA8EJAAAAAMpAcAIAAACAMnhYXQAAAAAskpsrpZ6QUo5JuU5zn2EUOMAoeV+59l/IsZfyfhdT26W8X4HbjXpIQQ0E90dwAgAAcFfZ6VLSESnpcN7XI+funzksJR+VnFlWV1mzefpLo96R2l5ndSWoZAQnAACAmig3V0pLPBeCCoai/NtpieV4IZsUEC55eBfYZSv8eLn3X8ixl7C/4K4Kf+0LODYtUTq5V/rkNmngU1K/KecdB3dCcAIAAKiOstOlpKMFgtD5X49KzsyyX8fTXwqJkoIb5W1ReVve/aAGksOz8j+PO3JmS98+Lq1/V1r2tHRilzR8puTpY3VlqAQEJwAAgKpmGOa1RedPoTsTe+GjRYGRBUJRXjAqGJR8QhgFqSwOT2n4y1JYOzNAbZ4vndwn3fKRFBBmdXWoYAQnAACAilZktKjgNLrDlzBa1KjAaFGUGZo8vCr/86B0vSZK9VpKn90hHVkrvXW1dOt8KbKz1ZWhAtkMo1C7ELeXnJys4OBgJSUlKSgoyOpyAABATWMYUmqilBRbTCjKu516ohwvVMJoUf7tkChGi2qaxL3S/NHmdU+eftLIt6R211tdFUpxIdmA4AQAAFBQdobZbS4/CJ05/9qiI+UfLSoYgs4PRoENGC1yR+mnpc/GS/tXmPev+ZvU/88E4GrqQrIBU/UAAEDt4RotKmkK3YWMFkWcN1rUuPB93zr8slwb+daRbl8oLZ0mrX1L+vHv0onfpRv+I3n6Wl0dLoHlwWnWrFn697//rbi4OHXo0EEzZ85U//79iz12/Pjxev/994vsb9++vbZv317ZpQIAgOru/NGi4tYwysko+3U8/QqPDp3fcIHRIpTG4SFd92+pfltpyaPS1s+kU/ulWz42AzdqJEun6i1YsEBjx47VrFmz1K9fP7355pt65513tGPHDjVu3LjI8UlJSUpPT3fdz8nJUZcuXfTggw/q6aefLtd7MlUPAAA3ZBjSsqekNf+RjNwyDi5utOi8qXSMFqGi7F8pfTpOyjgjBTU0w1ODrlZXhTw15hqn3r17q3v37po9e7ZrX7t27TRixAjNmDGjzOd/9dVXGjlypA4cOKAmTZqU6z0JTgAAuKGVL0jLnzNvFzdaVDAkBTVktAhV6+Q+af4tUuJuycNXGvmm1P5Gq6uCasg1TllZWdqwYYP+8pe/FNo/ePBgrVmzplyv8e6772rgwIGlhqbMzExlZp67gDM5OfniCgYAANXThrnnQtPQF6Re9zBahOqlXgtpQoz0+V3Svh/MEairnpCufIzv1RrEbtUbJyYmyul0Kjw8vND+8PBwxcfHl/n8uLg4ffvtt7r77rtLPW7GjBkKDg52bVFRUZdUNwAAqEZ+XyIteti83f/PUu97+UUU1ZNviHTbp1Lv+8z7K/5pBqns9FKfhurDsuCUz3be/9wMwyiyrzhz585VSEiIRowYUepx06ZNU1JSkms7fPjwpZQLAACqi9hfpc/vNK9p6jbGbPsMVGcOD2no89L1r0p2D2n7F9KcoVJynNWVoRwsC06hoaFyOBxFRpcSEhKKjEKdzzAMvffeexo7dqy8vEqfo+zt7a2goKBCGwAAqOESdkofjzY75LW+Vhr+KiNNqDmix0vj/k/yrSsd2yi9fbV09Derq0IZLAtOXl5eio6OVkxMTKH9MTEx6tu3b6nPXblypfbu3asJEyZUZokAAKA6SjoizRtldilr1FO6aY75l3ygJml6uTTxR7NleUqcNOc6adsXVleFUlg6VW/q1Kl655139N5772nnzp16+OGHFRsbq0mTJkkyp9mNGzeuyPPeffdd9e7dWx07dqzqkgEAgJXST5uhKfmoFNravGbEy8/qqoCLU7eZ2TSi1WApJ92cerp8hpRbVkt9WMHSP8+MHj1aJ0+e1LPPPqu4uDh17NhRS5YscXXJi4uLU2xsbKHnJCUlaeHChXr11VetKBkAAFglO12af6t04ncpMFIa84XkV9fqqoBL4xMk3fqJFPOk9Mvr0srnze/xEbP5o0A1Y+k6TlZgHScAAGogZ47ZwnnXYsk7WLrrWym8g9VVARXrtw/NLpG52VJkF+mW+VJwQ6urcmsXkg0s76oHAABQKsOQFk81Q5PDW7p1PqEJ7qn7WOmOryW/elLcZunta6QjG6yuCnkITgAAoHpb8bz02/uSzS6Nekdq2s/qioDK06Sv2TQirL10Nl6ae5209XOrq4IITpbKduZq1e4T2n4syepSAAConta9a17zIUnXvSi1v8HaeoCqUKepNOF7qfVQs+X+wgnSj/+gaYTFCE4WenHpLo17b63eXX3A6lIAAKh+dn4jLXnEvH3l41JPliFBLeIdKN3ykdTvT+b9Vf+WPhsnZaVaW1ctRnCy0KD25kK/MTuOKyPbaXE1AABUIwd/lj6fIBm5Uvc7pKumWV0RUPXsDmnQs2aHPYeX+ceE94aYa5mhyhGcLNS9cR1FBPkoJTNHq3afsLocAACqh+PbzbbjzkypzXXSsJclm83qqgDrdL1NumOR5F9fit8qvXW1dHid1VXVOgQnC9ntNg3rHClJWrQlzuJqAACoBs4clubdJGUmSVGXSTe9JzksXXYSqB4a9zabRoR3lFITpLnDpM0LrK6qViE4WWx4XnBatvO40rOYrgcAqMXSTknzRkopx6T6bc22456+VlcFVB8hjaW7lkpthpkjsl/eIy17mqYRVYTgZLGuUSFqGOKrtCynVuxKsLocAACskZUmfXyzlLhbCmoojVko+dW1uiqg+vEOkEbPky5/2Lz/0yvSgjFS5llr66oFCE4Ws9lsrlEnpusBAGolZ470+Z3SkXWST7AZmoIbWV0VUH3Z7dLAp6U/vGUuCr1rsdk04kys1ZW5NYJTNTC8cwNJ0g+/H1dqZo7F1QAAUIUMQ1r0J2n3d5KHj3Tbp1JYO6urAmqGLqOl8Ysl/zDp+Dbp7Wuk2P9ZXZXbIjhVAx0bBqlJPT9lZOfqh9+ZrgcAqEV+/Ie0cZ5ks0s3zZEaX2Z1RUDNEtXTbBoR0UlKPSG9P1zaNN/qqtwSwakaKDhdb/GWYxZXAwBAFfnfW9LqF83bw2dKba+ztBygxgqJMptGtB0uObOkryZJMU9KuTQeq0gEp2piWCdzut7yXSeUkpFtcTUAAFSy7V9K3z5m3r56uhR9h7X1ADWdl79084fSFY+a939+Vfrkdikzxdq63AjBqZpoFxmo5vX9lZWTq2U7j1tdDgAAlefAaumLeyQZUo+7zv2iB+DS2O3SNX+VRr1rNo3Y/a307mDp9CGrK3MLBKdqwpyuZ446LdpMdz0AgJuK3yp9cps5najd9dJ1L0o2m9VVAe6l003Snd9KAeFSwg7p7aulQ79YXVWNR3CqRvKvc1q154SS0pmuBwBwM6cPSfNGSZnJUpN+0sh3JLvD6qoA99QoWpq4XIrsIqWdlN6/3mzEgotGcKpGWocHqnV4gLKdhr7fHm91OQAAVJzURGneSOnscSmsvXTLx5Knj9VVAe4tuKF053dS+xFSbrb0fw9IS6fTNOIiEZyqGdd0PRbDBQC4i6xU6eObpZN7peAoc4Fb3xCrqwJqBy8/s9X/lX8x7//yujT/Fikj2dq6aiCCUzWTP13v572JOp2aZXE1AABcIme29Okd0tENkm8dacwXUlADq6sCahe7Xbp6mhmgPHykPd9L7w6STh2wurIaheBUzTSvH6D2kUHKyTW0lOl6AICazDCkrx+S9sZIHr7SbZ9J9VtbXRVQe3UcaTaNCIyUTvwuvX2NdPAnq6uqMQhO1dCwvFEnpusBAGq0ZU9Lmz+WbA7pj3OlqJ5WVwSgYXezaUSD7lL6KemDG6UNc62uqkYgOFVD1+dd57RmX6ISz2ZaXA0AABfh19nSzzPN2ze8JrW51tJyABQQFCnduUTqOErKzZG++ZP03TTJmWN1ZdUawakaalzPT50bBSvXkL7bxnQ9AEANs22h+UuYJF3zN6nbGGvrAVCUp6+5UO7V0837v86S5o+WMpKsrasaIzhVU8M65U/XO2ZxJQAAXID9K6Qv7pVkSL3ukfr/2eqKAJTEZpOufEy6+QPJ00/au0x6Z6B0cp/VlVVLBKdqKv86p/8dOKWE5AyLqwEAoBziNkufjDHXi2k/Qrr2efMXMwDVW/sbpbu+k4IaSom7pXcGSAdWWV1VtUNwqqYa1fFTt8YhMgxpyVaaRAAAqrlTB6R5N0lZKVLT/tIf3pTsDqurAlBekV2kiT9KDXtI6aelD/8grX/P6qqqFYJTNZa/GO5ighMAoDo7e0KaN1JKTZDCO0m3fCR5+lhdFYALFRghjV8sdfqj2TRi0cPSksdoGpGH4FSNXdcpQpK07uBpxSWlW1wNAADFyDwrffxH6dR+KaSxNOZzySfY6qoAXCxPH2nk22ZjF0la+6b00U3mKFQtR3CqxiKDfdWzaR1J0mLWdAIAVDc5WdKnY6VjGyW/etKYL82/WAOo2Ww26YpHpNHzzKYR+5ebTSMS91pdmaUITtVc/nQ9FsMFAFQrubnS/z0g7fvR/MXqts+k0JZWVwWgIrW7XrprqRTUSDq5V3rnGmnfcqursgzBqZob2ilCNpu06fAZHT6VZnU5AACYlj0pbf1UsnuYrYwbRVtdEYDKENlZume51KiXucbTvFHS2retrsoSBKdqLizQR72b1ZVEdz0AQDWx5nVpzX/M2ze8LrUaZG09ACpXQJh0xzdS51skwykteURaNFVyZltdWZUiONUATNcDAFQbWz6Vvp9u3h74jNT1VmvrAVA1PH2kP7whDXxakk1a/645+pR2yurKqgzBqQYY2jFCDrtNW48m6WBiqtXlAABqq70/SF/dZ96+7H6p35+srQdA1bLZpMsflm75WPL0lw6sNBfLPbHb6sqqBMGpBqgX4K2+LepJYk0nAIBFjm2UPh1nru3ScZQ0+DnzlygAtU/b66QJ30vBjc2lCN4ZaP5hxc0RnGqIYZ0iJTFdDwBggZP7pHk3SVlnpWZXSiNmS3Z+hQBqtYiO0sQfpajLpMwkc62n/70pGYbVlVUa/q9XQ1zbMUIedpt2xiVr34mzVpcDAKgtUo5L80ZKaYlSRGdzXRcPb6urAlAdBNSX7vha6nq7ZORK3z4mLXrYbZtGEJxqiBA/L13eKlQSi+ECAKpIRrL5V+TTB6U6TaUxCyWfIKurAlCdeHhLN/5XGvwPSTZpwxzpwz+4ZdMIglMNcq673jGLKwEAuL2cLGnBGCl+i+QXKo35wmxJDADns9mkvg9Kty2QvAKlg6ult6+RTuyyurIKRXCqQQa1D5eXw67dx89q9/EUq8sBALir3Fyze96BlWbnrNs/k+q1sLoqANVd6yHS3TFSSBPp9AGzacSeZVZXVWEITjVIsK+nrmhtTtdbtJlRJwBAJTAMc52mbZ9Ldg9p9IdSw+5WVwWgpghrJ01cLjXpJ2UmSx//Ufpllls0jSA41TCu6Xpb42S4wTcgAKCa+flV6ddZ5u0Rs6WWA6ytB0DN419PGvuV1G2s2TRi6TTpm4fMKcA1GMGphhnYPlxeHnbtP5GqnXFM1wMAVKBN86VlT5m3Bz8ndb7Z2noA1FweXtIN/5GGzJBsdum3D6QPR0ipJ62u7KIRnGqYAG8PXd2mviSaRAAAKtCeZdLXk83bfSZLfSdbWw+Ams9mk/rcL932qeQdJB36WXr7ailhp9WVXRTLg9OsWbPUrFkz+fj4KDo6WqtXry71+MzMTE2fPl1NmjSRt7e3WrRooffee6+Kqq0eznXXY7oeAKACHNkgfTpWys2ROt0sDfq71RUBcCetBkl3L5PqNJPOHJLeGSTtXmp1VRfM0uC0YMECTZkyRdOnT9fGjRvVv39/DR06VLGxsSU+5+abb9YPP/ygd999V7t27dL8+fPVtm3bKqzaegPahcnH067YU2nadjTZ6nIAADVZ4l7z4u3sNKnFNeZ6LHbL/64KwN3UbyNN/FFq2l/KSpE+Hi1tW2h1VRfEZlg4ZNG7d291795ds2fPdu1r166dRowYoRkzZhQ5/rvvvtMtt9yi/fv3q27duhf1nsnJyQoODlZSUpKCgmruIn4PfPSbFm+N071XNNe069pZXQ4AoCZKiZfeHSSdiZUadJPuWCR5B1hdFQB35syWljwi7fvR7L7nH2ppOReSDSz7k1JWVpY2bNigwYMHF9o/ePBgrVmzptjnfP311+rRo4deeOEFNWzYUK1bt9Yjjzyi9PT0Et8nMzNTycnJhTZ3MLxzpCSm6wEALlJGkjTvJjM01W0u3fYZoQlA5XN4SsNnSvestDw0XSgPq944MTFRTqdT4eHhhfaHh4crPj6+2Ofs379fP/30k3x8fPTll18qMTFR999/v06dOlXidU4zZszQM888U+H1W+3qtmHy93Lo6Jl0bTx8Rt0b17G6JABATZGTKX1yu3R8q+QfJo35Qgqob3VVAGoLm03yu7jZY1ayfBKzzWYrdN8wjCL78uXm5spms+mjjz5Sr169dN111+nll1/W3LlzSxx1mjZtmpKSklzb4cOHK/wzWMHH06GB7c3QuXhLnMXVAABqjFyn9MU90sHVklegNOZzqW4zq6sCgGrPsuAUGhoqh8NRZHQpISGhyChUvsjISDVs2FDBwcGufe3atZNhGDpy5Eixz/H29lZQUFChzV0M62RO11u8JU65uUzXAwCUwTCk7/4i7fhKsntKt8yTIrtYXRUA1AiWBScvLy9FR0crJiam0P6YmBj17du32Of069dPx44d09mzZ137du/eLbvdrkaNGlVqvdXRlW3qK9DbQ/HJGdoQe9rqcgAA1d1PL0tr3zJv/+ENqflVlpYDADWJpVP1pk6dqnfeeUfvvfeedu7cqYcfflixsbGaNGmSJHOa3bhx41zH33bbbapXr57uvPNO7dixQ6tWrdKjjz6qu+66S76+vlZ9DMt4ezg0qIM5OrdoM4vhAgBKsXGe9MOz5u1rn5c63WRtPQBQw1ganEaPHq2ZM2fq2WefVdeuXbVq1SotWbJETZo0kSTFxcUVWtMpICBAMTExOnPmjHr06KHbb79d119/vV577TWrPoLlrs9bDHfJtng5ma4HACjOru+krx8yb/ebIl12n6XlAEBNZOk6TlZwl3Wc8mXl5Krnc8uUlJ6t+RMvU58W9awuCQBQnRxeJ71/vZSTLnW5TRoxy+xoBQCoGes4oWJ4edg1JH+63ham6wEACjixW/r4j2ZoajlIuuE1QhMAXCSCkxsYnjdd77tt8cpx5lpcDQCgWkg+Js0bKaWflhpGSze/by48CQC4KAQnN9C3RT3V8fPUydQs/br/lNXlAACsln5GmjdKSjos1Wsp3faZ5OVvdVUAUKMRnNyAh8OuazuaazoxXQ8AarnsDOmT26SEHVJAhDTmC8mf618B4FIRnNzE9Z3N4PTd9nhlM10PAGqnXKf0xd3SoZ8l7yBpzOdSnSZWVwUAboHg5CZ6N6+n0ABvnUnL1s97E60uBwBQ1QxDWvKotPMbyeEl3fKxFNHJ6qoAwG0QnNyEw27TdZ0iJEmLtsRZXA0AoMqt+re0/l1JNmnk21Kz/lZXBABuheDkRoZ1MqfrLd0er8wcp8XVAACqzIb3peXPmbev+7fUYYSl5QCAOyI4uZGeTesqPMhbKRk5Wr2b6XoAUCv8vkRaNMW83f8RqddES8sBAHdFcHIjdrtN1+WNOi3eynQ9AHB7sb9Kn98pGblStzHSNX+1uiIAcFsEJzczPK+7XsyO48rIZroeALithJ3Sx6OlnAyp9bXS8Fclm83qqgDAbRGc3Ey3qDpqEOyjs5k5WrHrhNXlAKjtcp1mtzdUrKQj5gK3GWekRr2km+ZIDg+rqwIAt8b/Zd2M3W7TsM6Renv1AS3ackzXdoywuiQAtYFhSEmHpePbz20JO6TEPZKHtxTUQApqKAU3yvvaUApqlPe1oeQTZPUnqDnST5uhKfmoFNpaum2B5OVndVUA4PYITm5oeOcGenv1Af2wM0HpWU75ejmsLgmAO0k/Y4ai/HB0fLs5bSwzufjjs9Okk3vNrSTewedC1PmhKriRGbw8fSvl49Qo2enS/FulE79LgZHSmC8kv7pWVwUAtQLByQ11bhSsqLq+OnwqXT/+nqBhedc9AcAFcWabI0bHt0sJ26XjeSEp+Ujxx9s9zRGQ8A5SeHspvKNUv63kzDJHR5KOms9NOlr4fkaSlJkkJSSZQawkfvVKHrUKbmQGCYdn5fxbVAfOHOnzCVLsL2bQHLNQComyuioAqDUITm7IZrNpWKcGemPlPi3acozgBKB0hiElH8sbPdp2LiAl7pZys4t/TlCjcwEprIN5u15LycOr+OPrtSj5/TPP5gWpIyUHrOxUKe2kucVvKeGFbFJgRCmjVg2lgHDJXgMv7zUMafFUaddiyeEt3Trf/DcHAFQZgpObGt45Um+s3Kcff0/Q2cwcBXhzqgFIykwxp9XlB6T8sJSRVPzxXoF5o0cdpLC8UaSwdpJvSMXV5B0g1W9jbsUxDLMJQtKRkketko+ZI1spceZ2dH3xr2X3kAIblDItMMqc+lbdutOteF767X3JZpdueldq2s/qigCg1uG3aTfVoUGQmoX660Biqn7YeVw3dm1odUkAqpIzRzq1r3CjhuPbpTOHij/e5pBCWxUISHmjSMFR1ocIm03yrWNuEZ2KPyY3V0pLLH3UKiVOys2RkmLNrSQePqWPWgU3lHyCK+ezFmfdu9LK583b170otbu+6t4bAOBCcHJT5nS9SL2+fK8WbYkjOAHuyjCks8eLBqQTuyRnZvHPCYwsHI7CO5jXJnl4V23tFclulwLCzK1h9+KPceaY/1ZJR84LVQXCVmqCuS7SqX3mVhKvwDKaWTSsmE53O7+Rljxi3r7ycannhEt/TQDARSE4ubHhXczgtHLXCSVnZCvIx40vmgZqg6xUKeH3vEYNBYJS2snij/f0N6fV5TdqyA9LtbULm8Mjr5FEQ0m9iz8mJ9Oc9ldaM4v001JWitnZ7sTvJb+fb93SR60CG5R8TZgkHfzZbAZh5ErR46Wrpl3KpwcAXCKCkxtrEx6olmEB2ptwVjHbj2tUdCOrSwJQHrlO6dSBAp3stpkB6dQBScUsJmuzS3VbnBeQ2kshTWtmIwQreXhLdZuZW0myUs1w5RqpOnLeFMGjUtZZKf2UucVvLeGFbOYIWXGjVh4+0peTzFHDNsOk616yfsokANRyBCc3ZrPZNLxzpGYu26PFW+MITkB1lJpYuJNdwnZzVCknvfjj/cMKd7ILb2+2/GaNo6rj5W9eDxbaqvjHDcNstlHsqFWBgOXMNKcOnj0uHfut+NeKusxsBuHgxzUAWI3/E7u5/OC0es8JJaVlK9iP6XqAJbLTzeuOXNch5YWl1ITij/fwMafZFQxIYR2kgPpVWzcunM1mdh30DSm5ZbhhmFMsizSzKBCyQhpLo+cRigGgmiA4ubmWYYFqGxGo3+NTtHR7vG7uyWKJQKXKzTU71xVs1HB8u9lowMgt5gk2c1pYwWYNYR3MfXZHlZePKmKzSf6h5tagq9XVAADKgeBUCwzvHKnf41O0aGscwQmoSGmnCoej49vNNZKyU4s/3rdugXCUfz1SW3PqFwAAqNYITrXAsM4N9OL3u/Xz3kSdSs1SXf9SujgBKMowzIAUv+1co4bj2811gYrj8DIXcy3YqCG8oxQQzgX+AADUUASnWqBZqL86NAjS9mPJ+m5bvG7r3djqkoCaI3GvtPhh6cCq4h8PaVw0INVtwcX8AAC4GX6y1xLDOzfQ9mPJWrTlGMEJKI+cTOnnV6VVL5rdzxzeUsPowo0awtpJPkFWVwoAAKoAwamWGN45Uv/67nf9uv+kTqRkqn6gt9UlAdXXwZ+lRVOkxN3m/RYDpGEvlb62DwAAcGusjFhLRNX1U5eoEOUa0nfbSrguA6jt0k5J//eANPc6MzT5h0mj3pXGLCQ0AQBQyxGcapHhnSIlSd9sITgBhRiGtPkT6fUe0sZ55r7oO6XJa6VON9HQAQAAEJxqk2GdzeC07uApHU/OsLgaoJo4uU/64Abpy3vNBUnrt5PuWipdP1PyrWN1dQAAoJogONUiDUJ8Fd2kjgxDWrKVUSfUcjmZ0soXpFl9zI55Hj7SgCele1dJjS+zujoAAFDNEJxqmWF50/UWMV0PtdmhNdIb/aXlz5kd81pcI93/i9T/z5IH65wBAICiCE61zLDOkbLZpA2HTuvYmXSrywGqVtop6f8mS3OGSom7JP/6ec0fvpDqNre6OgAAUI0RnGqZ8CAf9WxaV5K0mFEn1BaGIW1eIL3eU9r4obkverw0eR3NHwAAQLkQnGqh6/OaRCziOifUBif3SR/cKH15j5SWKNVvm9f84VWaPwAAgHIjONVC13aMlN0mbT58RodPpVldDlA5crKklf/Oa/6w0mz+cM3fpHtX0/wBAABcMIJTLVQ/0FuXNa8niSYRcFOH1khvXC4t/4fZ/KH51dJ9a6QrHqH5AwAAuCgEp1pqeOcGkqRFW45ZXAlQgdJOSV8/WLj5w8h3pLFfSvVaWF0dAACowQhOtdS1HSPksNu0/ViyDiSmWl0OcGkMQ9ryqdn84bcPzH3d75AeWCt1/iPNHwAAwCUjONVSdf291LeFOV1vMaNOqMlO7pM+HCF9MfFc84c7v5NueE3yq2t1dQAAwE0QnGqx613T9bjOCTVQTpa0Kq/5w/4VksNbuuavZvOHJn2srg4AALgZglMtNqRDhDwdNv0en6K9CSlWlwOU36FfpDf7Sz/mN3+4Srr/F+mKR2n+AAAAKgXBqRYL9vNU/1b1JTHqhBoi7ZT09UPSnGulE79LfqHSyLelsV/R/AEAAFQqy4PTrFmz1KxZM/n4+Cg6OlqrV68u8dgVK1bIZrMV2X7//fcqrNi9DOuUtxjuljgZhmFxNUAJDEPa8pn0317Sb++b+7qPkyavkzrfTPMHAABQ6TysfPMFCxZoypQpmjVrlvr166c333xTQ4cO1Y4dO9S4ceMSn7dr1y4FBQW57tevX78qynVLgzqEy+sLu/YmnNWu4ylqGxFU9pOAqnRqv7RoqrR/uXk/tI10/UypSV9LywIAALWLpSNOL7/8siZMmKC7775b7dq108yZMxUVFaXZs2eX+rywsDBFRES4NofDUUUVu58gH09d2cYMnouZrofqJCdLWvViXvOH5Wbzh6v/Kk36idAEAACqnGXBKSsrSxs2bNDgwYML7R88eLDWrFlT6nO7deumyMhIDRgwQMuXLy/12MzMTCUnJxfaUNjwzkzXQzXjav7wdyknQ2p2pdn84UqaPwAAAGtYFpwSExPldDoVHh5eaH94eLji4+OLfU5kZKTeeustLVy4UF988YXatGmjAQMGaNWqVSW+z4wZMxQcHOzaoqKiKvRzuIMB7cLl7WHXgcRUbT9GsISF0k9L3/ypQPOHetIf3pLG/R/NHwAAgKUsvcZJkmznXdRtGEaRffnatGmjNm3auO736dNHhw8f1osvvqgrrrii2OdMmzZNU6dOdd1PTk4mPJ0nwNtD17QN07fb4rVoS5w6Ngy2uiTUNoYhbVsoffcXKfWEua/bWGnQsyxiCwAAqgXLRpxCQ0PlcDiKjC4lJCQUGYUqzWWXXaY9e/aU+Li3t7eCgoIKbShqeN5iuIu3HmO6HqrWqf3SvJHSwglmaAptLY1fIt34OqEJAABUG5YFJy8vL0VHRysmJqbQ/piYGPXtW/4Lvzdu3KjIyMiKLq/WuaZtmHw9HTp8Kl1bjiRZXQ5qg5wsafVLZvOHfT/mNX+YbjZ/aNrP6uoAAAAKsXSq3tSpUzV27Fj16NFDffr00VtvvaXY2FhNmjRJkjnN7ujRo/rggw8kSTNnzlTTpk3VoUMHZWVlad68eVq4cKEWLlxo5cdwC75eDg1oF6ZFW+K0aMsxdYkKsbokuLPY/0mLpkgJO8z7za6Qhr0ihba0tCwAAICSWBqcRo8erZMnT+rZZ59VXFycOnbsqCVLlqhJkyaSpLi4OMXGxrqOz8rK0iOPPKKjR4/K19dXHTp00OLFi3XddddZ9RHcyvDODbRoS5wWb4nTtKHtZLezqCgqWPppadkz0oY55n2/etKQf0qdR7OILQAAqNZsRi27oCU5OVnBwcFKSkrieqfzZGQ7Ff33GKVmObXwvr6KblLH6pLgLlzNH6ZJqQnmvm5jpEF/5zomAABgmQvJBpYugIvqxcfToUHtzcYci7Ycs7gauI1TB6R5o/KaPyTkNX9YLN34X0ITAACoMQhOKCS/u96SrXHKza1Vg5GoaM5safXL0qzLpH0/SA6vAs0fLre6OgAAgAti+TpOqF76tw5VoI+Hjidnat3BU+rdvJ7VJaEmOrzWXMg2v/lD0/7S8Jk0fwAAADUWI04oxNvDoSEdIiRJi7fGWVwNapz0M9Kih6V3B5mhybeuNOIN6Y5vCE0AAKBGIzihiGGdzXWxlmyNl5PpeiiP/OYPr/eU1r9n7us6Rpq8Xup6Kx3zAABAjcdUPSvFb5O2fCKFd5LCO5gXzXt4WV2VLm8ZqhA/TyWezdT/9p9U35ahVpeE6uz0QWnxn6W9y8z79VpJw1+RmvW3tCwAAICKRHCy0qGfpTX/OXff7inVb2OGKNfWSQoIq9K/2Hs67Lq2Q4Q+WXdY32yJIziheM5s6ZfXpRX/knLSzeYP/f8sXf6w5OFtdXUAAAAViuBkpfCOUs+J0vHt5paZJB3fZm4F+YWaISqi07lAFdpG8vSptNKGd26gT9Yd1nfb4vT3GzvIw8GsThRweK30zRQpYbt5v2l/c5QptJWlZQEAAFQWgpOVmvYzN8m8RiTpcF6I2mZO4zu+XTq1T0pLlA6sNLd8Noc5tS8/SOWHqsDIChmduqx5XdXz99LJ1Cyt2XdSV7Suf8mvCTeQfkb64dm865gMs/nDkOekLlzHBAAA3BvBqbqw2aSQxubWZui5/Vlp0onf80ai8kam4rdKGWekEzvNbdvn5473rWOOZIV3PBeqwtpJnr4XVI6Hw65rO0boo//FatGWYwSn2s4wpO1fSt/9RTp73NzX9XZp0N8lf1rWAwAA92czDKNWtU1LTk5WcHCwkpKSFBQUZHU5F8cwpORj50an8kNV4h7JcBY93maX6rUscN1UXrAKblTqKMEv+07q1rd/VbCvp9ZNHygvD6br1UqnD0qLH5H2xpj367XMa/5whaVlAQAAXKoLyQaMONVENpsU3NDcWg8+tz87I290anvhUJV2UkrcbW7bvzx3vHdw3jS/jucCVVg7yctfktSrWV3VD/TWiZRM/bw3UVe3DaviDwpLObOlX/4rrXie5g8AAKDWIzi5E08fqUFXc8tnGObUqoLXTR3fLiXuMptRxK4xNxebVLe5FN5BjvCOerhRkGb97qNFm48QnGqTw+ukRVPONSppcrk5ylS/taVlAQAAWIWperVVTpYZns5vRpGaUOzhZw1f+UV1kt3V2a+jFN5e8g6s4sJRqTKSpGXPFGj+UEca/JzU9TaaPwAAALfDVD2UzcPL7MQX0anw/rMJhRpRGMe3KTt+pwJs6dKRteZWUJ2mBRpR5H2t00yycz1UjWIY0o6vpG8fP9f8octt0uC/S/6s4wUAAMCIE8r03NdbtPKXNRrTNEXjmqecm+6XElf8Ezz9zWulIs7r7ucTXLWFo3xOH5KWPCLt+d68X7eFdP1Mmj8AAAC3x4gTKtTQrlF6e02U/nXUoZvvGiQfT4f5QOrJwm3Sj2+TEnZK2anS0fXmVlBw4wLrTuWFqrrNJbuj6j8UzOYPv86Sls8wmz/YPaX+U6XLp1bq4soAAAA1EcEJZeoWFaKGIb46eiZdy39P0NBOkeYD/vWk5leaWz5njrlo7/nNKJKPSEmx5rb723PHe/iao1MFp/qFd5D86lbth6xtjqyXvvlTgeYP/aThM2n+AAAAUAKCE8pks9k0vHOk3ly1X4u2xp0LTsVxeEj125hbx1Hn9qefLjwyFZ83OpWTLh37zdwKCmp4XpjqaK4f5Kil37KGkbc5pVxnga+55lZkX0mP50gbP5LWvaNzzR/+YS5mS/MHAACAEtXS30JxoYblBacfdyYoLStHfl4X+K3jW0dqerm55ct1SqcOSMe3Fg5VZ2Kl5KPmln/djSQ5vKWwtufClH/9EgLD+cGhwO0i+5xmICn0OvnHnn9cbumvk5tbzD5n0cBT5HXOu53/vuc/XtG63GqGJpo/AAAAlInghHLp1DBYjev6KfZUmn7YmaDruzS49Be1O6TQlubW4Q/n9mckScd3nFvA9/h28352qhS32dxQlM0u2Rzmv6vrdjH7giKlAU8VnmIJAACAUhGcUC750/VmrdinRVuOVUxwKolPsNSkj7nly82VzhwscN3UNinrrBkEbPa8YJAfFAqGhfyvtmL22c97bhmBw17wvarhezDVDgAAoNIQnFBuwzs30KwV+7R81wmdzcxRgHcVfvvY7WYHvrrNpfY3VN37AgAAAJJYpRTl1i4yUM1D/ZWVk6tlO45bXQ4AAABQZQhOKLf86XqStGjLMYurAQAAAKoOwQkXZHjetU0rd59QUnq2xdUAAAAAVYPghAvSOjxQrcMDlO00FMN0PQAAANQSBCdcsGGdzFEnpusBAACgtiA44YIN72Je5/TTnkSdTs2yuBoAAACg8hGccMFa1A9Qu8gg5eQaWro93upyAAAAgEpHcMJFye+ut3hrnMWVAAAAAJWP4ISLkh+c1uw7qZNnMy2uBgAAAKhcBCdclCb1/NWpYbCcuYa+3cZ0PQAAALg3ghMummu63ham6wEAAMC9EZxw0YblBaf/HTiphJQMi6sBAAAAKg/BCRetUR0/dY0KUa4hfbuV6XoAAABwXwQnXJL86XoshgsAAAB3RnDCJcmfrrfu4GnFJzFdDwAAAO6J4IRLEhnsq55N60hiTScAAAC4L4ITLtmwTkzXAwAAgHsjOOGSXdcpUjabtDH2jI6cTrO6HAAAAKDCEZxwycKCfNS7WV1J0hKm6wEAAMANEZxQIYZ1biBJWsRiuAAAAHBDBCdUiKEdI2S3SVuOJOnQyVSrywEAAAAqFMEJFSI0wFt9W4RKYtQJAAAA7sfy4DRr1iw1a9ZMPj4+io6O1urVq8v1vJ9//lkeHh7q2rVr5RaIcstfDHcxwQkAAABuxtLgtGDBAk2ZMkXTp0/Xxo0b1b9/fw0dOlSxsbGlPi8pKUnjxo3TgAEDqqhSlMeQDhHysNu0Iy5Z+0+ctbocAAAAoMJYGpxefvllTZgwQXfffbfatWunmTNnKioqSrNnzy71effee69uu+029enTp4oqRXnU8fdSv5ZM1wMAAID7sSw4ZWVlacOGDRo8eHCh/YMHD9aaNWtKfN6cOXO0b98+PfXUU+V6n8zMTCUnJxfaUHnyp+uxGC4AAADciWXBKTExUU6nU+Hh4YX2h4eHKz4+vtjn7NmzR3/5y1/00UcfycPDo1zvM2PGDAUHB7u2qKioS64dJRvcIUKeDpt2Hz+r3cdTrC4HAAAAqBCWN4ew2WyF7huGUWSfJDmdTt1222165pln1Lp163K//rRp05SUlOTaDh8+fMk1o2TBvp66olV9SUzXAwAAgPuwLDiFhobK4XAUGV1KSEgoMgolSSkpKVq/fr0mT54sDw8PeXh46Nlnn9XmzZvl4eGhH3/8sdj38fb2VlBQUKENlWt4l3PT9QzDsLgaAAAA4NJZFpy8vLwUHR2tmJiYQvtjYmLUt2/fIscHBQVp69at2rRpk2ubNGmS2rRpo02bNql3795VVTrKMLBduLw87Np/IlW/xzNdDwAAADVf+S4UqiRTp07V2LFj1aNHD/Xp00dvvfWWYmNjNWnSJEnmNLujR4/qgw8+kN1uV8eOHQs9PywsTD4+PkX2w1qBPp66uk19Ld1+XIu2HFO7SEb5AAAAULNZGpxGjx6tkydP6tlnn1VcXJw6duyoJUuWqEmTJpKkuLi4Mtd0QvU0rHODvOAUp0cGtyn2ujUAAACgprAZtewilOTkZAUHByspKYnrnSpRamaOov8Ro4zsXH0z+XJ1ahRsdUkAAABAIReSDSzvqgf35O/toQFtzSYfi7ayphMAAABqNoITKk3+YriLt8TRXQ8AAAA1GsEJleaqNmHy83LoyOl0bTp8xupyAAAAgItGcEKl8fVyaGC7vOl6LIYLAACAGozghEqVP11vydY45eYyXQ8AAAA1E8EJleqK1vUV6O2huKQM/RZ72upyAAAAgItCcEKl8vF0aFB7pusBAACgZiM4odIN75LXXW9rnJxM1wMAAEANRHBCpbu8ZX0F+XjoREqm1h08ZXU5AAAAwAUjOKHSeXnYNaRDhCRp0RYWwwUAAEDNQ3BClRjepYEk6dut8cpx5lpcDQAAAHBhCE6oEn1b1FMdP0+dTM3Sr/uZrgcAAICa5aKC0+HDh3XkyBHX/bVr12rKlCl66623KqwwuBdPh13XdsxvEsF0PQAAANQsFxWcbrvtNi1fvlySFB8fr0GDBmnt2rV64okn9Oyzz1ZogXAf+YvhfrstXtlM1wMAAEANclHBadu2berVq5ck6dNPP1XHjh21Zs0affzxx5o7d25F1gc30rtZXYUGeOlMWrZ+3ptodTkAAABAuV1UcMrOzpa3t7ckadmyZbrhhhskSW3btlVcHIucongeDruG5k3XYzFcAAAA1CQXFZw6dOigN954Q6tXr1ZMTIyuvfZaSdKxY8dUr169Ci0Q7iV/ut7S7fHKymG6HgAAAGqGiwpO//rXv/Tmm2/qqquu0q233qouXbpIkr7++mvXFD6gOD2a1lVYoLdSMnK0es8Jq8sBAAAAysXjYp501VVXKTExUcnJyapTp45r/z333CM/P78KKw7ux2G36bpOkZq75qAWbYnTgHbhVpcEAAAAlOmiRpzS09OVmZnpCk2HDh3SzJkztWvXLoWFhVVogXA/13cxp+vF7DiujGynxdUAAAAAZbuo4HTjjTfqgw8+kCSdOXNGvXv31ksvvaQRI0Zo9uzZFVog3E+3qDpqEOyjs5k5Wrmb6XoAAACo/i4qOP3222/q37+/JOnzzz9XeHi4Dh06pA8++ECvvfZahRYI92PPm64n0V0PAAAANcNFBae0tDQFBgZKkr7//nuNHDlSdrtdl112mQ4dOlShBcI9De/SQJL0w87jSs9iuh4AAACqt4sKTi1bttRXX32lw4cPa+nSpRo8eLAkKSEhQUFBQRVaINxTl0bBiqrrq7Qsp5bvSrC6HAAAAKBUFxWcnnzyST3yyCNq2rSpevXqpT59+kgyR5+6detWoQXCPdlsNg3rZI46LdpyzOJqAAAAgNJdVHC66aabFBsbq/Xr12vp0qWu/QMGDNArr7xSYcXBveUvhvvj7wlKzcyxuBoAAACgZBcVnCQpIiJC3bp107Fjx3T06FFJUq9evdS2bdsKKw7urUODIDWt56eM7Fwt23nc6nIAAACAEl1UcMrNzdWzzz6r4OBgNWnSRI0bN1ZISIj+/ve/Kzc3t6JrhJuy2Wwa3tmcrreY7noAAACoxjwu5knTp0/Xu+++q+eff179+vWTYRj6+eef9fTTTysjI0PPPfdcRdcJNzWsc6ReX75XK3afUEpGtgJ9PK0uCQAAACjiooLT+++/r3feeUc33HCDa1+XLl3UsGFD3X///QQnlFvbiEC1qO+vfSdSFbPjuEZ2b2R1SQAAAEARFzVV79SpU8Vey9S2bVudOnXqkotC7VFwuh6L4QIAAKC6uqjg1KVLF73++utF9r/++uvq3LnzJReF2iW/u97qPSeUlJZtcTUAAABAURc1Ve+FF17QsGHDtGzZMvXp00c2m01r1qzR4cOHtWTJkoquEW6uVXig2oQHatfxFC3dEa+be0RZXRIAAABQyEWNOF155ZXavXu3/vCHP+jMmTM6deqURo4cqe3bt2vOnDkVXSNqgfxRJ6brAQAAoDqyGYZhVNSLbd68Wd27d5fT6ayol6xwycnJCg4OVlJSkoKCgqwuB3kOJKbq6hdXyGG3ad30garr72V1SQAAAHBzF5INLnoBXKAiNQv1V4cGQXLmGlq6Pd7qcgAAAIBCCE6oNoa5pusds7gSAAAAoDCCE6qN4Z3MtuS/7DupEymZFlcDAAAAnHNBXfVGjhxZ6uNnzpy5lFpQyzWu56cujYK1+UiSvtser7GXNbG6JAAAAEDSBQan4ODgMh8fN27cJRWE2m145wbafCRJizYfIzgBAACg2rig4ESrcVS26zpH6rklO7X24CkdT85QeJCP1SUBAAAAXOOE6qVhiK+6Nw6RYUhLtrKmEwAAAKoHghOqneGdzSYRi1kMFwAAANUEwQnVznWdImWzSesPndaxM+lWlwMAAABYH5xmzZqlZs2aycfHR9HR0Vq9enWJx/7000/q16+f6tWrJ19fX7Vt21avvPJKFVaLqhAR7KOeTepKYroeAAAAqgdLg9OCBQs0ZcoUTZ8+XRs3blT//v01dOhQxcbGFnu8v7+/Jk+erFWrVmnnzp3661//qr/+9a966623qrhyVLbhXczFcL9huh4AAACqAZthGIZVb967d291795ds2fPdu1r166dRowYoRkzZpTrNUaOHCl/f399+OGH5To+OTlZwcHBSkpKUlBQ0EXVjcp3IiVTvf+5TLmGtPqxqxVV18/qkgAAAOBmLiQbWDbilJWVpQ0bNmjw4MGF9g8ePFhr1qwp12ts3LhRa9as0ZVXXlkZJcJC9QO9dVnzepKkxUzXAwAAgMUsC06JiYlyOp0KDw8vtD88PFzx8fGlPrdRo0by9vZWjx499MADD+juu+8u8djMzEwlJycX2lAzDOtsTtdbtOWYxZUAAACgtrO8OYTNZit03zCMIvvOt3r1aq1fv15vvPGGZs6cqfnz55d47IwZMxQcHOzaoqKiKqRuVL6hHSPlsNu07WiyDiamWl0OAAAAajHLglNoaKgcDkeR0aWEhIQio1Dna9asmTp16qSJEyfq4Ycf1tNPP13isdOmTVNSUpJrO3z4cEWUjypQ199LfVswXQ8AAADWsyw4eXl5KTo6WjExMYX2x8TEqG/fvuV+HcMwlJmZWeLj3t7eCgoKKrSh5hieN13vm81M1wMAAIB1PKx886lTp2rs2LHq0aOH+vTpo7feekuxsbGaNGmSJHO06OjRo/rggw8kSf/973/VuHFjtW3bVpK5rtOLL76oBx980LLPgMo1pEOEpn+5Tb/Hp2hvwlm1DAuwuiQAAADUQpYGp9GjR+vkyZN69tlnFRcXp44dO2rJkiVq0qSJJCkuLq7Qmk65ubmaNm2aDhw4IA8PD7Vo0ULPP/+87r33Xqs+AipZiJ+X+rcK1fJdJ7RoyzFNGdja6pIAAABQC1m6jpMVWMep5lm44Yj+/NlmtQoLUMxUWs8DAACgYtSIdZyA8hrUIVxeDrv2JJzVrvgUq8sBAABALURwQrUX5OOpK1rXl8SaTgAAALAGwQk1wvVd8hfDjVMtm10KAACAaoDghBphQLtweXvYdSAxVTvikq0uBwAAALUMwQk1QoC3h65uEybJHHUCAAAAqhLBCTXGcNd0vWNM1wMAAECVIjihxrimbZh8PR06fCpdW48mWV0OAAAAahGCE2oMPy8PDWjHdD0AAABUPYITapThnc3peovprgcAAIAqRHBCjXJVmzD5ezl09Ey6fos9Y3U5AAAAqCUITqhRfDwdGtQ+XJI56gQAAABUBYITapzhnRtIkr7adFTHzqRbXA0AAABqA4ITapwrWtdXm/BAnUrN0vg5a5WUnm11SQAAAHBzBCfUOF4edr13Z0+FB3lr9/GzmvThBmXmOK0uCwAAAG6M4IQaqWGIr94b31P+Xg79sv+kHv98C132AAAAUGkITqixOjQI1uwx0fKw2/TVpmP699JdVpcEAAAAN0VwQo12Rev6mjGykyRp1op9mvfrIYsrAgAAgDsiOKHG+2OPKD08sLUk6cn/26ZlO45bXBEAAADcDcEJbuGhAS01ukeUcg3pwfkbtfnwGatLAgAAgBshOMEt2Gw2/eMPHXVF6/pKz3ZqwvvrFHsyzeqyAAAA4CYITnAbng67Zt3eXR0aBCnxbJbumLNWp1KzrC4LAAAAboDgBLcS4O2hOeN7qmGIrw4kpuru99cpI5s1ngAAAHBpCE5wO2FBPpp7Z08F+Xjot9gzmvLJJjlzWeMJAAAAF4/gBLfUKjxQb43rIS+HXd9tj9c/Fu+wuiQAAADUYAQnuK3LmtfTizd3kSTN+fmg3lm93+KKAAAAUFMRnODWbujSQNOGtpUkPbdkp5ZsjbO4IgAAANREBCe4vXuuaK5xfZrIMKQpCzZp3cFTVpcEAACAGobgBLdns9n01PUdNKh9uLJycnX3++u1N+Gs1WUBAACgBiE4oVZw2G167ZZu6hoVoqT0bI2fs1YJKRlWlwUAAIAaguCEWsPXy6F37+ihpvX8dOR0uibMXa/UzByrywIAAEANQHBCrVIvwFtz7+yluv5e2no0SZM//k05zlyrywIAAEA1R3BCrdM01F/v3NFDPp52Ld91Qn/7v+0yDBbIBQAAQMkITqiVujeuo1dv6SabTZq/NlazVuyzuiQAAABUYwQn1FpDOkTo6es7SJL+vXSXvvjtiMUVAQAAoLoiOKFWu6NvU917RXNJ0mOfb9HPexMtrggAAADVEcEJtd7j17bV9V0aKCfX0KQPN+j3+GSrSwIAAEA1Q3BCrWe32/TiHzurV7O6SsnM0fj31ikuKd3qsgAAAFCNEJwASd4eDr09todahgUoPjlDd85Zp+SMbKvLAgAAQDVBcALyBPt5au6dPVU/0Fu/x6fovnkblJXDGk8AAAAgOAGFNKrjpznje8rfy6Gf957UXxZuYY0nAAAAEJyA83VsGKxZY6LlsNv0xcajeun73VaXBAAAAIsRnIBiXNm6vmb8oZMk6fXle/Xx/2ItrggAAABWIjgBJbi5Z5QeGtBKkvS3/9um5b8nWFwRAAAArEJwAkrx8MBWuim6kZy5hu7/6DdtOXLG6pIAAABgAYITUAqbzaYZIzupf6tQpWc7ddfcdTp8Ks3qsgAAAFDFLA9Os2bNUrNmzeTj46Po6GitXr26xGO/+OILDRo0SPXr11dQUJD69OmjpUuXVmG1qI08HXbNur272kUGKfFslu6Ys1anU7OsLgsAAABVyNLgtGDBAk2ZMkXTp0/Xxo0b1b9/fw0dOlSxscVfiL9q1SoNGjRIS5Ys0YYNG3T11Vfr+uuv18aNG6u4ctQ2gT7mGk8Ngn20/0SqJn6wXhnZTqvLAgAAQBWxGRYuUtO7d291795ds2fPdu1r166dRowYoRkzZpTrNTp06KDRo0frySefLNfxycnJCg4OVlJSkoKCgi6qbtReu4+naNTsNUrJyNGwTpH6z63dZLfbrC4LAAAAF+FCsoFlI05ZWVnasGGDBg8eXGj/4MGDtWbNmnK9Rm5urlJSUlS3bt0Sj8nMzFRycnKhDbhYrcMD9ebYaHk6bFq8NU7/XLLT6pIAAABQBSwLTomJiXI6nQoPDy+0Pzw8XPHx8eV6jZdeekmpqam6+eabSzxmxowZCg4Odm1RUVGXVDfQt0WoXvxjF0nSOz8d0Hs/HbC4IgAAAFQ2y5tD2GyFpzkZhlFkX3Hmz5+vp59+WgsWLFBYWFiJx02bNk1JSUmu7fDhw5dcM3Bj14Z6/Nq2kqS/L96hb7fGWVwRAAAAKpOHVW8cGhoqh8NRZHQpISGhyCjU+RYsWKAJEybos88+08CBA0s91tvbW97e3pdcL3C+SVc219EzaZr3a6ymLNiksCBvRTcpedooAAAAai7LRpy8vLwUHR2tmJiYQvtjYmLUt2/fEp83f/58jR8/Xh9//LGGDRtW2WUCJbLZbHr6+g4a2C5MmTm5uvv99dp/4qzVZQEAAKASWDpVb+rUqXrnnXf03nvvaefOnXr44YcVGxurSZMmSTKn2Y0bN851/Pz58zVu3Di99NJLuuyyyxQfH6/4+HglJSVZ9RFQy3k47Hrt1m7q0ihYp9OyNX7OOiWezbS6LAAAAFQwS4PT6NGjNXPmTD377LPq2rWrVq1apSVLlqhJkyaSpLi4uEJrOr355pvKycnRAw88oMjISNf2pz/9yaqPAMjPy0Pvju+pxnX9FHsqTRPmrlNaVo7VZQEAAKACWbqOkxVYxwmVZf+Jsxo1e41Op2VrQNswvTk2Wh4Oy/uvAAAAoAQ1Yh0nwN00rx+gd+7oIW8Pu374PUFPfb1dtezvEgAAAG6L4ARUoOgmdfXqLd1ks0kf/S9Ws1fus7okAAAAVACCE1DBru0YoSeHt5ckvfDdLv3fpqMWVwQAAIBLRXACKsGd/Zrp7subSZIe+Wyz1uxLtLgiAAAAXAqCE1BJnriunYZ1ilS209C9H27QrvgUq0sCAADARSI4AZXEbrfppZu7qGfTOkrJyNGdc9YqPinD6rIAAABwEQhOQCXy8XTo7XE91KK+v44lZWj8nLVKyci2uiwAAABcIIITUMlC/Lw0985eCg3w1u/xKbr/o9+U7cy1uiwAAABcAIITUAWi6vppzvie8vNyaPWeRP1l4VbWeAIAAKhBCE5AFenUKFj/va27HHabFv52RK8s22N1SQAAACgnghNQha5uG6Z/jOgoSXrthz1asC7W4ooAAABQHgQnoIrd2quxHrympSTpiS+3afmuBIsrAgAAQFkIToAFpg5qrZHdG8qZa+iBj37TtqNJVpcEAACAUhCcAAvYbDY9P7KzLm8ZqrQsp+6cu06HT6VZXRYAAABKQHACLOLlYdesMd3VNiJQJ1IydefcdUpKY40nAACA6ojgBFgoyMdTc+7sqYggH+1NOKuJH65XZo7T6rIAAABwHoITYLHIYF/NvaunAr09tPbAKf35083KzWWNJwAAgOqE4ARUA20jgvTm2Gh5OmxatCVOz3/3u9UlAQAAoACCE1BN9G0Zqhdu6ixJemvVfr2/5qC1BQEAAMCF4ARUI3/o1kiPDmkjSXr6m+1auj3e4ooAAAAgEZyAauf+q1ro1l6NZRjSQ/M36rfY01aXBAAAUOsRnIBqxmaz6e83dtA1bcOUmZOru99fr4OJqVaXBQAAUKsRnIBqyMNh139u7abOjYJ1KjVLd8xZq5NnM60uCwAAoNYiOAHVlL+3h969o6ei6vrq0Mk03fX+eqVnscYTAACAFQhOQDVWP9Bbc+/spRA/T20+fEYPfbJRTtZ4AgAAqHIEJ6Caa1E/QO+M6yEvD7tidhzXM99sl2EQngAAAKoSwQmoAXo0rauZo7vKZpM++OWQ3lq13+qSAAAAahWCE1BDXNcpUtOvaydJmvHt7/p68zGLKwIAAKg9CE5ADXJ3/+a6q18zSdIjn27Wr/tPWlwRAABA7UBwAmqYvw5rp6EdI5TlzNU9H6zXnuMpVpcEAADg9ghOQA1jt9v0yuiuim5SR8kZORo/Z52OJ2dYXRYAAIBbIzgBNZCPp0PvjOuh5qH+OnomXXfOWaezmTlWlwUAAOC2CE5ADVXH30tz7+yl0AAv7YhL1v0f/aZsZ67VZQEAALglghNQgzWu56d37+gpX0+HVu0+oelfbmWNJwAAgEpAcAJquC5RIXr9tm6y26RP1x/Rqz/ssbokAAAAt0NwAtzAgHbh+vuIjpKkmcv26NP1hy2uCAAAwL0QnAA3cXvvJrr/qhaSpCe+2KpVu09YXBEAAID7IDgBbuTRIW00omsD5eQaum/eBm0/lmR1SQAAAG6B4AS4EZvNphdu6qI+zespNcupO+es09Ez6VaXBQAAUOMRnAA34+Vh1xtjo9UmPFAJKZka/95aJaVlW10WAABAjUZwAtxQsK+n5tzZUxFBPtqTcFb3fLhemTlOq8sCAACosQhOgJtqEOKrOXf2VIC3h/534JQe/WyLcnNZ4wkAAOBiEJwAN9YuMkizx3SXh92mrzcf0wtLd1ldEgAAQI1EcALcXP9W9fX8qM6SpDdW7tOHvxy0tiAAAIAaiOAE1AI3RTfSnwe1liT97f+2694P12tnXLLFVQEAANQclgenWbNmqVmzZvLx8VF0dLRWr15d4rFxcXG67bbb1KZNG9ntdk2ZMqXqCgVquMnXtNTE/s1ks0lLtx/X0FdX6/6PNmhXfIrVpQEAAFR7lganBQsWaMqUKZo+fbo2btyo/v37a+jQoYqNjS32+MzMTNWvX1/Tp09Xly5dqrhaoGaz2WyaPqy9vp9yhYZ1jpTNJi3ZGq9rX12lBz7+TXuOE6AAAABKYjMMw7I2W71791b37t01e/Zs17527dppxIgRmjFjRqnPveqqq9S1a1fNnDnzgt4zOTlZwcHBSkpKUlBQ0MWUDbiFXfEpevWH3VqyNV6SZLNJ13duoIcGtFLLsACLqwMAAKh8F5INLBtxysrK0oYNGzR48OBC+wcPHqw1a9ZU2PtkZmYqOTm50AZAahMRqFm3R+vbP/XXkA7hMgzp683HNPiVlXp4wSbtP3HW6hIBAACqDcuCU2JiopxOp8LDwwvtDw8PV3x8fIW9z4wZMxQcHOzaoqKiKuy1AXfQLjJIb47tocUPXa5B7cOVa0hfbjyqgS+v1NRPN+lgYqrVJQIAAFjO8uYQNput0H3DMIrsuxTTpk1TUlKSazt8+HCFvTbgTjo0CNbb43rom8mXa0DbMOUa0he/HdWAl1fq0c82K/ZkmtUlAgAAWMbDqjcODQ2Vw+EoMrqUkJBQZBTqUnh7e8vb27vCXg9wd50aBevd8T21+fAZzVy2W8t3ndBnG47oy41HNap7I02+pqWi6vpZXSYAAECVsmzEycvLS9HR0YqJiSm0PyYmRn379rWoKgD5ukSFaM6dvfTl/X11Rev6ysk1tGD9YV394gpN+2Krjp5Jt7pEAACAKmPZiJMkTZ06VWPHjlWPHj3Up08fvfXWW4qNjdWkSZMkmdPsjh49qg8++MD1nE2bNkmSzp49qxMnTmjTpk3y8vJS+/btrfgIgNvr1riOPrirlzYcOq2Zy3Zr9Z5EzV8bq883HNbNPaL0wNUt1SDE1+oyAQAAKpWl7cglcwHcF154QXFxcerYsaNeeeUVXXHFFZKk8ePH6+DBg1qxYoXr+OKuf2rSpIkOHjxYrvejHTlwadYdPKVXYnZrzb6TkiQvh1239IrS/Ve1VESwj8XVAQAAlN+FZAPLg1NVIzgBFeN/+0/qlWW79ev+U5IkLw+7buvVWPdd1ULhQQQoAABQ/RGcSkFwAirWmn2JmhmzR2sPmgHK28Ou23s30aSrmisskAAFAACqL4JTKQhOQMUzDENr9p3UKzG7tf7QaUmSj6ddY3o30b1XtlD9QDpbAgCA6ofgVAqCE1B5DMPQ6j2JemXZbm2MPSNJ8vV0aFyfJrrniuaqF0CAAgAA1QfBqRQEJ6DyGYahlbtP6JWY3dp8JEmS5Ofl0Lg+TXXPFc1V19/L4goBAAAITqUiOAFVxzAMLd+VoFdi9mjrUTNA+Xs5NL5fU03s31whfgQoAABgHYJTKQhOQNUzDEPLdiZo5rLd2n4sWZIU4O2hu/o11YTLmyvYz9PiCgEAQG1EcCoFwQmwjmEY+n7Hcc1ctkc748wAFejjobv6NdNdlzdTsC8BCgAAVB2CUykIToD1cnMNLd0er5nL9mjX8RRJUpCPh+7u31x39muqQB8CFAAAqHwEp1IQnIDqIzfX0Lfb4jVz2W7tSTgrSQr29dTE/s00vl8zBXh7WFwhAABwZwSnUhCcgOrHmWto8dY4vbpst/adSJUk1fHz1MQrmuuOPk3lT4ACAACVgOBUCoITUH05cw0t2nJMry7bo/2JZoCq6++le65ornF9msjPiwAFAAAqDsGpFAQnoPrLcebq683H9NoPe3TwZJokKTTAS/de0UJjLmsiXy+HxRUCAAB3QHAqBcEJqDlynLn6apMZoGJP5Qcob913VQvd3ruxfDwJUAAA4OIRnEpBcAJqnmxnrr787ahe+3GPjpxOlySFBZoB6tZeBCgAAHBxCE6lIDgBNVe2M1cLNxzRf37cq6NnzAAVHuStB65uqdE9o+TtQYACAADlR3AqBcEJqPmycnL12YbD+u+Pe3UsKUOSFBnso/uvbqmbezQiQAEAgHIhOJWC4AS4j8wcpz5df0T//XGv4pPNANUg2EeTr2mlm6IbycvDbnGFAACgOiM4lYLgBLifjGynFqw7rFkr9up4cqYkqWGIrx68pqVGRTeSp4MABQAAiiI4lYLgBLivjGyn5q+N1awV+3QixQxQUXV99eA1rTSyW0N5EKAAAEABBKdSEJwA95eR7dS8Xw/pjZX7lHg2S5LUpJ6fHrymlUZ0bUCAAgAAkghOpSI4AbVHWlaO5v16SG+u3K+TqWaAahbqr4cGtNQNXRrKYbdZXCEAALASwakUBCeg9knNzNEHvxzSW6v26XRatiSpeX1//WlAKw3v3IAABQBALUVwKgXBCai9zmbm6P01B/X26v06kxegWoYF6E8DWmlYp0jZCVAAANQqBKdSEJwApGRka+7PZoBKzsiRJLUOD9CUga11bYcIAhQAALUEwakUBCcA+ZIzsjXnp4N656f9SskLUG0jAjVlYCsNbk+AAgDA3RGcSkFwAnC+pPRsvfvTAc356YBSMs0A1T4ySFMGttKg9uGy2QhQAAC4I4JTKQhOAEpyJi3LDFA/H9TZvADVsWGQpgxorQHtwghQAAC4GYJTKQhOAMpyOjVLb6/er7lrDiotyylJ6tQwWFe0DlWbiCC1jQhUs1B/ebIeFAAANRrBqRQEJwDldSo1S2+t2q/31xxUeraz0GOeDpta1A9Qm4hAcws3vzYM8WVkCgCAGoLgVAqCE4ALdfJsphZvjdPOuGTtik/R7uNnXVP5zhfo7aHWEYFqHR6otgVCVR1/ryquGgAAlIXgVAqCE4BLZRiGjpxO1674FO06nmJ+jU/RvhNnlZNb/P9SwwK91SbCDFNmqApSq/AA+Xg6qrh6AACQj+BUCoITgMqSlZOrA4mp+j0+f2QqRb/Hp+jI6fRij7fbpKb1/NU6b5pf/ghVk3r+ctAKHQCASkdwKgXBCUBVS8nI1u7jZ7U7b3QqP1idTssu9nhvD7tahQeoTbjZiKJ1XqgKC/Tm+ikAACoQwakUBCcA1YFhGDpxNtM1ze/3vBGq3cdTlJGdW+xzQvw8C1071TYiUK3CAxXk41nF1QMA4B4ITqUo7z+O0+lUdnbxfw0GLpanp6ccDq5pQcmcuYZiT6W5AtWu48n6PT5FBxNTVcLlU2oY4luku1+L+gHy8qBdOgAApSE4laKsfxzDMBQfH68zZ85UfXGoFUJCQhQREcGUK1yQjGyn9iacLdKQIj45o9jjPew2Na/vX2CEKkhtwgPVqI6v7Fw/BQCAJIJTqcr6x4mLi9OZM2cUFhYmPz8/frlFhTEMQ2lpaUpISFBISIgiIyOtLglu4ExaVqFGFPnBKiWj+Hbp/l4OtQov2N3PHKGqF+BdxZUDAGC9CwlOHlVUU43gdDpdoalevXpWlwM35OvrK0lKSEhQWFgY0/ZwyUL8vNS7eT31bn7u/1mGYSguKaPQtVO/x6doX8JZpWY5tenwGW06fKbQ64QGeKtNxLmGFG0iAtUqPEB+XvyYAABAIjgVkn9Nk5+fn8WVwJ3lf39lZ2cTnFApbDabGoT4qkGIr65uG+ban+3M1cHEVNdUv/wRqthTaUo8m6nEvZn6ee/JAq8jNa7r57puKr8hRdN6/vJwcP0UAKB2ITgVg+l5qEx8f8Eqng67WoWbnfiGdz63PzUzR3sSzmpXfLJrhGpXfIoSz2bp0Mk0HTqZpu93HHcd7+Wwq0VYgGtkKr8pRWSwD9/fAAC3RXACgFrO39tDXaNC1DUqpND+xLOZ2l1gZOr34ynaczxFaVlO7YxL1s645ELHB/l4nNfdz2xIEexHu3QAQM1HcEKJrrrqKnXt2lUzZ84s1/EHDx5Us2bNtHHjRnXt2rVSawNQ+UIDvBXa0lt9W4a69uXmGjpyOt21iG/+tL/9ialKzsjRuoOnte7g6UKvU9ffS43q+OZtfoVuNwzxlb83P4oAANUfP63cQFlTY+644w7NnTv3gl/3iy++kKdn+f9SHBUVpbi4OIWGhpZ98CUgoAHWsdttalzPT43r+WlwhwjX/swcp/YlpBbo7pes3cfP6uiZdJ1KzdKp1CxtOZJU7GsSrAAANQE/jdxAXFyc6/aCBQv05JNPateuXa59+Z3c8mVnZ5crENWtW/eC6nA4HIqIiCj7QABux9vDofYNgtS+QeFWrikZ2Tp8Kl1HTqfpyOn0vC3N9TU5I6fMYFXHz7NIoCr4lWAFAKgK/LQpg2EYSs92WvLevp6Ocl1oXTCsBAcHy2azufYdPHhQkZGRWrBggWbNmqVff/1Vs2fP1g033KDJkydr9erVOnXqlFq0aKEnnnhCt956q+u1zp+q17RpU91zzz3au3evPvvsM9WpU0d//etfdc8997jeq+BI0IoVK3T11Vdr2bJlevzxx7Vjxw517dpVc+bMUZs2bVzv849//EOvvfaa0tPTNXr0aIWGhuq7777Tpk2bLurfLTMzU48++qg++eQTJScnq0ePHnrllVfUs2dPSdLp06c1efJkff/99zp79qwaNWqkJ554QnfeeaeysrI0depULVy4UKdPn1ZERITuvfdeTZs27aJqAWq7QB9PtW/gWSRQ5UtKz9bR06UHq9Np2TqdlqStRy88WDWs46sAghUAoAJY/tNk1qxZ+ve//624uDh16NBBM2fOVP/+/Us8fuXKlZo6daq2b9+uBg0a6LHHHtOkSZMqrb70bKfaP7m00l6/NDueHVJha6g8/vjjeumllzRnzhx5e3srIyND0dHRevzxxxUUFKTFixdr7Nixat68uXr37l3i67z00kv6+9//rieeeEKff/657rvvPl1xxRVq27Ztic+ZPn26XnrpJdWvX1+TJk3SXXfdpZ9//lmS9NFHH+m5557TrFmz1K9fP33yySd66aWX1KxZs4v+rI899pgWLlyo999/X02aNNELL7ygIUOGaO/evapbt67+9re/aceOHfr2228VGhqqvXv3Kj09XZL02muv6euvv9ann36qxo0b6/Dhwzp8+PBF1wKgdMG+ngr2LTlYJWfkB6vCgSo/ZCWlZ19UsGoY4qtGdc3bBCsAQHlY+tNiwYIFmjJliuuX5jfffFNDhw7Vjh071Lhx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KVN05syZSn0fAADKw8NhV5DDrqAKDGPZzlzXFMP8EHY2s5iRL1dAOzdiluXMVVaOuWU7c5WZk+val13gsSxnrrKdhdf2ysk1lJPlVFqWs8I+S0U4F8CKGSk7b3Qt/3Hvch5//micp+t5thJfmxG6mo3g5AYmTJigkSNH6tChQ2rSpEmhx9577z117dr1gkOTJNWvX7+iSixTRERElb0XAADuytNhV7CvXcG+FRfGimMYRqGgleXMVXaOoSyn0wxcOWa4Mh9z5n01zh2f4zQfzw9oxYSzQq+f//j5x5wX7s4PdNlOQ9lOp1KrWaBz2AsHOu8Cgcvz/JDmKBjKzhuVK+b4gqN7niWEusL7CHXlRXAqi2FI2WnWvLenn1SOIebhw4crLCxMc+fO1VNPPeXan5aWpgULFuif//ynTp48qcmTJ2v16tU6deqUWrRooSeeeEK33npria97/lS9PXv2aMKECVq7dq2aN2+uV199tchzHn/8cX355Zc6cuSIIiIidPvtt+vJJ5+Up6en5s6dq2eeeUaSXEPnc+bM0fjx44tM1du6dav+9Kc/6ZdffpGfn59GjRqll19+WQEB5kXB48eP15kzZ3T55ZfrpZdeUlZWlm655RbNnDlTnp4X98MiNjZWDz74oH744QfZ7XZde+21+s9//qPw8HBJ0ubNmzVlyhStX79eNptNrVq10ptvvqkePXro0KFDmjx5sn766SdlZWWpadOm+ve//63rrrvuomoBAKA6s9ls8vZwyNujei1KnJtruEJXdk7hoJVZMJwVCFuZhcLf+eGscPjLD4PnRuOcrn3FvbbrOc7cQnU6cw2l5zpdHSCrEw+7rVCoKhjSiky7dBQembvQENi7WV3V8fey+iOXG8GpLNlp0j8bWPPeTxyTvMqeKufh4aFx48Zp7ty5evLJJ12h5LPPPlNWVpZuv/12paWlKTo6Wo8//riCgoK0ePFijR07Vs2bN1fv3r3LfI/c3FyNHDlSoaGh+vXXX5WcnFzstU+BgYGaO3euGjRooK1bt2rixIkKDAzUY489ptGjR2vbtm367rvvXNMKg4ODi7xGWlqarr32Wl122WVat26dEhISdPfdd2vy5MmaO3eu67jly5crMjJSy5cv1969ezV69Gh17dpVEydOLPPznM8wDI0YMUL+/v5auXKlcnJydP/992v06NFasWKFJOn2229Xt27dNHv2bDkcDm3atMkV0h544AFlZWVp1apV8vf3144dO1whDwAAVA273SYfu0M+ntUr0OWP0JUUsooPXkaR0brs80JZ5vn7nLnKyjHKDHUFQ2Kx0y6rKNQtvK+voglOqGp33XWX/v3vf2vFihW6+uqrJZnT9EaOHKk6deqoTp06euSRR1zHP/jgg/ruu+/02WeflSs4LVu2TDt37tTBgwfVqFEjSdI///lPDR06tNBxf/3rX123mzZtqj//+c9asGCBHnvsMfn6+iogIEAeHh6lTs376KOPlJ6erg8++MB1jdXrr7+u66+/Xv/6179cI0B16tTR66+/LofDobZt22rYsGH64YcfLio4LVu2TFu2bNGBAwcUFRUlSfrwww/VoUMHrVu3Tj179lRsbKweffRRtW3bVpLUqlUr1/NjY2M1atQoderUSZLUvHnzC64BAAC4p3MjdJK8ra6msJJCXZFQdn7wct0+L5wVGMkrGuryjzdDXbBvzYoiNataK3j6mSM/Vr13ObVt21Z9+/bVe++9p6uvvlr79u3T6tWr9f3330uSnE6nnn/+eS1YsEBHjx5VZmamMjMzy938YefOnWrcuLErNElSnz59ihz3+eefa+bMmdq7d6/Onj2rnJwcBQUFlftz5L9Xly5dCtXWr18/5ebmateuXa7g1KFDBzkc5/6iFBkZqa1bt17QexV8z6ioKFdokqT27dsrJCREO3fuVM+ePTV16lTdfffd+vDDDzVw4ED98Y9/VIsWLSRJDz30kO677z59//33GjhwoEaNGqXOnTtfVC0AAABVpTqHuuqG1dHKYrOZ0+Ws2C6wheaECRO0cOFCJScna86cOWrSpIkGDBggSXrppZf0yiuv6LHHHtOPP/6oTZs2aciQIcrKyirXaxuGUWTf+S0+f/31V91yyy0aOnSoFi1apI0bN2r69Onlfo+C71VS+9CC+8+/lslmsyk3N/f8p1zSexbc//TTT2v79u0aNmyYfvzxR7Vv315ffvmlJOnuu+/W/v37NXbsWG3dulU9evTQf/7zn4uqBQAAANUPwcmN3HzzzXI4HPr444/1/vvv684773T90r969WrdeOONGjNmjLp06aLmzZtrz5495X7t9u3bKzY2VseOnRt9++WXXwod8/PPP6tJkyaaPn26evTooVatWunQoUOFjvHy8pLTWfqc2fbt22vTpk1KTU0t9Np2u12tW7cud80XIv/zHT582LVvx44dSkpKUrt27Vz7WrdurYcffljff/+9Ro4cqTlz5rgei4qK0qRJk/TFF1/oz3/+s95+++1KqRUAAABVj+DkRgICAjR69Gg98cQTOnbsmMaPH+96rGXLloqJidGaNWu0c+dO3XvvvYqPjy/3aw8cOFBt2rTRuHHjtHnzZq1evVrTp08vdEzLli0VGxurTz75RPv27dNrr73mGpHJ17RpUx04cECbNm1SYmKiMjMzi7zX7bffLh8fH91xxx3atm2bli9frgcffFBjx451TdO7WE6nU5s2bSq07dixQwMHDlTnzp11++2367ffftPatWs1btw4XXnllerRo4fS09M1efJkrVixQocOHdLPP/+sdevWuULVlClTtHTpUh04cEC//fabfvzxx0KBCwAAADUbwcnNTJgwQadPn9bAgQPVuHFj1/6//e1v6t69u4YMGaKrrrpKERERrtbf5WG32/Xll18qMzNTvXr10t13363nnnuu0DE33nijHn74YU2ePFldu3bVmjVr9Le//a3QMaNGjdK1116rq6++WvXr19f8+fOLvJefn5+WLl2qU6dOqWfPnrrppps0YMAAvf766xf2j1GMs2fPqlu3boW26667TjabTV999ZXq1KmjK664QgMHDlTz5s21YMECSZLD4dDJkyc1btw4tW7dWjfffLOGDh3qaq/udDr1wAMPqF27drr22mvVpk0bzZo165LrBQAAQPVgM4q7eMWNJScnKzg4WElJSUWaFmRkZOjAgQNq1qyZfHx8LKoQ7o7vMwAAgOqhtGxwPkacAAAAAKAMBCcAAAAAKAPBCQAAAADKQHACAAAAgDIQnIpRy/ploIrx/QUAAFDzEJwK8PT0lCSlpaVZXAncWf73V/73GwAAAKo/D6sLqE4cDodCQkKUkJAgyVxPyGazWVwV3IVhGEpLS1NCQoJCQkLkcDisLgkAAADlRHA6T0REhCS5whNQ0UJCQlzfZwAAAKgZCE7nsdlsioyMVFhYmLKzs60uB27G09OTkSYAAIAaiOBUAofDwS+4AAAAACTRHAIAAAAAykRwAgAAAIAyEJwAAAAAoAy17hqn/MVHk5OTLa4EAAAAgJXyM0F+RihNrQtOKSkpkqSoqCiLKwEAAABQHaSkpCg4OLjUY2xGeeKVG8nNzdWxY8cUGBjI4raXIDk5WVFRUTp8+LCCgoKsLgeVjPNdu3C+axfOd+3DOa9dON+lMwxDKSkpatCggez20q9iqnUjTna7XY0aNbK6DLcRFBTEf4S1COe7duF81y6c79qHc167cL5LVtZIUz6aQwAAAABAGQhOAAAAAFAGghMuire3t5566il5e3tbXQqqAOe7duF81y6c79qHc167cL4rTq1rDgEAAAAAF4oRJwAAAAAoA8EJAAAAAMpAcAIAAACAMhCcAAAAAKAMBCe4rFq1Stdff70aNGggm82mr776qtDjhmHo6aefVoMGDeTr66urrrpK27dvL3RMZmamHnzwQYWGhsrf31833HCDjhw5UoWfAuU1Y8YM9ezZU4GBgQoLC9OIESO0a9euQsdwzt3H7Nmz1blzZ9cCiH369NG3337repxz7d5mzJghm82mKVOmuPZxzt3H008/LZvNVmiLiIhwPc65dk9Hjx7VmDFjVK9ePfn5+alr167asGGD63HOe8UjOMElNTVVXbp00euvv17s4y+88IJefvllvf7661q3bp0iIiI0aNAgpaSkuI6ZMmWKvvzyS33yySf66aefdPbsWQ0fPlxOp7OqPgbKaeXKlXrggQf066+/KiYmRjk5ORo8eLBSU1Ndx3DO3UejRo30/PPPa/369Vq/fr2uueYa3Xjjja4fopxr97Vu3Tq99dZb6ty5c6H9nHP30qFDB8XFxbm2rVu3uh7jXLuf06dPq1+/fvL09NS3336rHTt26KWXXlJISIjrGM57JTCAYkgyvvzyS9f93NxcIyIiwnj++edd+zIyMozg4GDjjTfeMAzDMM6cOWN4enoan3zyieuYo0ePGna73fjuu++qrHZcnISEBEOSsXLlSsMwOOe1QZ06dYx33nmHc+3GUlJSjFatWhkxMTHGlVdeafzpT38yDIP/vt3NU089ZXTp0qXYxzjX7unxxx83Lr/88hIf57xXDkacUC4HDhxQfHy8Bg8e7Nrn7e2tK6+8UmvWrJEkbdiwQdnZ2YWOadCggTp27Og6BtVXUlKSJKlu3bqSOOfuzOl06pNPPlFqaqr69OnDuXZjDzzwgIYNG6aBAwcW2s85dz979uxRgwYN1KxZM91yyy3av3+/JM61u/r666/Vo0cP/fGPf1RYWJi6deumt99+2/U4571yEJxQLvHx8ZKk8PDwQvvDw8Ndj8XHx8vLy0t16tQp8RhUT4ZhaOrUqbr88svVsWNHSZxzd7R161YFBATI29tbkyZN0pdffqn27dtzrt3UJ598ot9++00zZswo8hjn3L307t1bH3zwgZYuXaq3335b8fHx6tu3r06ePMm5dlP79+/X7Nmz1apVKy1dulSTJk3SQw89pA8++EAS/41XFg+rC0DNYrPZCt03DKPIvvOV5xhYa/LkydqyZYt++umnIo9xzt1HmzZttGnTJp05c0YLFy7UHXfcoZUrV7oe51y7j8OHD+tPf/qTvv/+e/n4+JR4HOfcPQwdOtR1u1OnTurTp49atGih999/X5dddpkkzrW7yc3NVY8ePfTPf/5TktStWzdt375ds2fP1rhx41zHcd4rFiNOKJf87jzn/wUiISHB9deMiIgIZWVl6fTp0yUeg+rnwQcf1Ndff63ly5erUaNGrv2cc/fj5eWlli1bqkePHpoxY4a6dOmiV199lXPthjZs2KCEhARFR0fLw8NDHh4eWrlypV577TV5eHi4zhnn3D35+/urU6dO2rNnD/99u6nIyEi1b9++0L527dopNjZWEj/DKwvBCeXSrFkzRUREKCYmxrUvKytLK1euVN++fSVJ0dHR8vT0LHRMXFyctm3b5joG1YdhGJo8ebK++OIL/fjjj2rWrFmhxznn7s8wDGVmZnKu3dCAAQO0detWbdq0ybX16NFDt99+uzZt2qTmzZtzzt1YZmamdu7cqcjISP77dlP9+vUrsoTI7t271aRJE0n8DK80FjSkQDWVkpJibNy40di4caMhyXj55ZeNjRs3GocOHTIMwzCef/55Izg42Pjiiy+MrVu3GrfeeqsRGRlpJCcnu15j0qRJRqNGjYxly5YZv/32m3HNNdcYXbp0MXJycqz6WCjBfffdZwQHBxsrVqww4uLiXFtaWprrGM65+5g2bZqxatUq48CBA8aWLVuMJ554wrDb7cb3339vGAbnujYo2FXPMDjn7uTPf/6zsWLFCmP//v3Gr7/+agwfPtwIDAw0Dh48aBgG59odrV271vDw8DCee+45Y8+ePcZHH31k+Pn5GfPmzXMdw3mveAQnuCxfvtyQVGS74447DMMwW1s+9dRTRkREhOHt7W1cccUVxtatWwu9Rnp6ujF58mSjbt26hq+vrzF8+HAjNjbWgk+DshR3riUZc+bMcR3DOXcfd911l9GkSRPDy8vLqF+/vjFgwABXaDIMznVtcH5w4py7j9GjRxuRkZGGp6en0aBBA2PkyJHG9u3bXY9zrt3TN998Y3Ts2NHw9vY22rZta7z11luFHue8VzybYRiGNWNdAAAAAFAzcI0TAAAAAJSB4AQAAAAAZSA4AQAAAEAZCE4AAAAAUAaCEwAAAACUgeAEAAAAAGUgOAEAAABAGQhOAAAAAFAGghMAoMZJSEjQvffeq8aNG8vb21sREREaMmSIfvnlF0mSzWbTV199ZW2RAAC34mF1AQAAXKhRo0YpOztb77//vpo3b67jx4/rhx9+0KlTp6wuDQDgphhxAgDUKGfOnNFPP/2kf/3rX7r66qvVpEkT9erVS9OmTdOwYcPUtGlTSdIf/vAH2Ww2131J+uabbxQdHS0fHx81b95czzzzjHJyclyP22w2zZ49W0OHDpWvr6+aNWumzz77zPV4VlaWJk+erMjISPn4+Khp06aaMWNGVX10AICFCE4AgBolICBAAQEB+uqrr5SZmVnk8XXr1kmS5syZo7i4ONf9pUuXasyYMXrooYe0Y8cOvfnmm5o7d66ee+65Qs//29/+plGjRmnz5s0aM2aMbr31Vu3cuVOS9Nprr+nrr7/Wp59+ql27dmnevHmFghkAwH3ZDMMwrC4CAIALsXDhQk2cOFHp6enq3r27rrzySt1yyy3q3LmzJHPk6Msvv9SIESNcz7niiis0dOhQTZs2zbVv3rx5euyxx3Ts2DHX8yZNmqTZs2e7jrnsssvUvXt3zZo1Sw899JC2b9+uZcuWyWazVc2HBQBUC4w4AQBqnFGjRunYsWP6+uuvNWTIEK1YsULdu3fX3LlzS3zOhg0b9Oyzz7pGrAICAjRx4kTFxcUpLS3NdVyfPn0KPa9Pnz6uEafx48dr06ZNatOmjR566CF9//33lfL5AADVD8EJAFAj+fj4aNCgQXryySe1Zs0ajR8/Xk899VSJx+fm5uqZZ57Rpk2bXNvWrVu1Z88e+fj4lPpe+aNL3bt314EDB/T3v/9d6enpuvnmm3XTTTdV6OcCAFRPBCcAgFto3769UlNTJUmenp5yOp2FHu/evbt27dqlli1bFtns9nM/Dn/99ddCz/v111/Vtm1b1/2goCCNHj1ab7/9thYsWKCFCxfSzQ8AagHakQMAapSTJ0/qj3/8o+666y517txZgYGBWr9+vV544QXdeOONkqSmTZvqhx9+UL9+/eTt7a06deroySef1PDhwxUVFaU//vGPstvt2rJli7Zu3ap//OMfrtf/7LPP1KNHD11++eX66KOPtHbtWr377ruSpFdeeUWRkZHq2rWr7Ha7PvvsM0VERCgkJMSKfwoAQBUiOAEAapSAgAD17t1br7zyivbt26fs7GxFRUVp4sSJeuKJJyRJL730kqZOnaq3335bDRs21MGDBzVkyBAtWrRIzz77rF544QV5enqqbdu2uvvuuwu9/jPPPKNPPvlE999/vyIiIvTRRx+pffv2rvf+17/+pT179sjhcKhnz55asmRJoRErAIB7oqseAAB5iuvGBwCAxDVOAAAAAFAmghMAAAAAlIFrnAAAyMPsdQBASRhxAgAAAIAyEJwAAAAAoAwEJwAAAAAoA8EJAAAAAMpAcAIAAACAMhCcAAAAAKAMBCcAAAAAKAPBCQAAAADKQHACAAAAgDL8P/abUdYBvNXYAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "class CausalSft_Model_C(nn.Module):\n",
    "    def __init__(self, base_model, num_labels, max_words):\n",
    "        super().__init__()\n",
    "\n",
    "        self.max_words = max_words\n",
    "        self.base_model = copy.deepcopy(base_model)\n",
    "        self.sft_model = copy.deepcopy(base_model)\n",
    "        self.sft_classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "        self.r_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 2, self.base_model.config.hidden_size // 2),\n",
    "        )\n",
    "        self.c_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "        )\n",
    "        self.num_patches = 10\n",
    "        self.patches_size = self.max_words // self.num_patches\n",
    "        self.patch_extractor = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches, self.base_model.config.hidden_size * self.num_patches // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches // 2, self.base_model.config.hidden_size),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size),\n",
    "        )\n",
    "        # self.c_patch_mixer = nn.Sequential(\n",
    "        #     nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 8),\n",
    "        #     nn.ReLU(),\n",
    "        #     nn.Linear(self.base_model.config.hidden_size // 8, self.base_model.config.hidden_size // 8),\n",
    "        # )\n",
    "        self.classifier = nn.Linear(self.base_model.config.hidden_size // 4, num_labels)\n",
    "\n",
    "        self.cross_entropy_loss = nn.CrossEntropyLoss()\n",
    "        self.mse_loss = nn.MSELoss()\n",
    "\n",
    "        for param in self.base_model.parameters():\n",
    "            param.requires_grad = False\n",
    "\n",
    "        print(f'Initialise Causal SFT model \"{model_name}\" (unfreezed all layers) with a linear head!')\n",
    "        count_parameters(self)\n",
    "\n",
    "    def entropy_maximization(self, feature):\n",
    "        p = F.softmax(feature, dim=-1)\n",
    "        log_p = F.log_softmax(feature, dim=-1)\n",
    "        entropy = -torch.mean(torch.mean(p * log_p, dim=-1))  # Maximize entropy\n",
    "        return -entropy  # Minimize negative entropy\n",
    "\n",
    "    def forward(self, input_ids1, attention_mask1, input_ids2, attention_mask2, input_ids3, attention_mask3, input_ids4, attention_mask4, labels=None):\n",
    "        B, _ = input_ids1.size()\n",
    "        # avoid unwanted computational graph\n",
    "        outputs_sft2 = self.sft_model(input_ids2, attention_mask=attention_mask2)[1]\n",
    "        outputs_sft2_logits = self.sft_classifier(outputs_sft2)\n",
    "        # pull out R0 and R1 using another data point\n",
    "        outputs_base3 = self.base_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        outputs_sft3 = self.sft_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        # optimise C from R0 and R1\n",
    "        base_C3= self.r_2_c(outputs_base3)[:, :self.base_model.config.hidden_size//4]\n",
    "        sft_C3 = self.r_2_c(outputs_sft3)[:, :self.base_model.config.hidden_size//4]\n",
    "        # gather C \n",
    "        outputs_sft = self.sft_model(input_ids1, attention_mask=attention_mask1)[1]\n",
    "        sft_C = self.r_2_c(outputs_sft)[:, :self.base_model.config.hidden_size//4]\n",
    "        # extract non-contextual embeddings\n",
    "        with torch.no_grad():\n",
    "            embeddings_1 = self.sft_model.embeddings(input_ids1, attention_mask1)[:, :self.max_words]\n",
    "            embeddings_2 = self.sft_model.embeddings(input_ids2, attention_mask2)[:, :self.max_words]\n",
    "            embeddings_4 = self.sft_model.embeddings(input_ids4, attention_mask4)[:, :self.max_words]\n",
    "\n",
    "        # input = embeddings_1\n",
    "        # patches = torch.sum(input.view(B, self.num_patches, self.patches_size, -1), dim=2)\n",
    "        # Ai_samples = self.patch_extractor(patches.view(B, -1))\n",
    "        # Ai_samples = Ai_samples[torch.randperm(B).to(input_ids1.device), :]\n",
    "        Ai_Z1 = self.c_2_c(sft_C)\n",
    "        logits = self.classifier(Ai_Z1)   \n",
    "\n",
    "        loss = None\n",
    "        if labels is not None:\n",
    "            loss_sft = self.cross_entropy_loss(outputs_sft2_logits, labels) \n",
    "            loss_identify = self.mse_loss(base_C3, sft_C3) + self.entropy_maximization(sft_C3) + self.entropy_maximization(base_C3)\n",
    "            loss_cls = self.cross_entropy_loss(logits, labels) \n",
    "            \n",
    "            loss = loss_sft + loss_identify + loss_cls \n",
    "        \n",
    "        return (loss, logits) if loss is not None else logits\n",
    "    \n",
    "set_seed(data_seed)\n",
    "model = CausalSft_Model_C(base_model, num_cls, max_words)\n",
    "\n",
    "# Define training arguments and trainer\n",
    "training_args = TrainingArguments(\n",
    "    overwrite_output_dir=True,\n",
    "    output_dir=f'./results/yelp/causal-sft-c/{N}-shot-{data_seed}',\n",
    "    eval_strategy=\"steps\",\n",
    "    save_strategy=\"steps\",\n",
    "    logging_strategy=\"steps\",\n",
    "    logging_steps=0.1,\n",
    "    save_steps=0.1,\n",
    "    learning_rate=5e-5,\n",
    "    per_device_train_batch_size=128,\n",
    "    per_device_eval_batch_size=128,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=True,\n",
    "    metric_for_best_model=\"eval_loss\",  # Choose model based on validation loss\n",
    "    greater_is_better=False,  # Lower validation loss is better\n",
    "    save_total_limit=1,\n",
    ")\n",
    "\n",
    "\n",
    "trainer = Trainer(\n",
    "    model=model,\n",
    "    args=training_args,\n",
    "    train_dataset=paired_train_dataset,\n",
    "    eval_dataset=paired_validation_dataset,\n",
    "    tokenizer=tokenizer,\n",
    "    data_collator=paired_data_collator,\n",
    "    compute_metrics=compute_metrics,\n",
    ")\n",
    "\n",
    "# Train the model\n",
    "trainer.train()\n",
    "\n",
    "results = {}\n",
    "train_results = trainer.evaluate(paired_train_dataset)\n",
    "print(f'train: {train_results}')\n",
    "results['train'] = train_results\n",
    "\n",
    "valid_results = trainer.evaluate(paired_validation_dataset)\n",
    "print(f'validation: {valid_results}')\n",
    "results['valid'] = valid_results\n",
    "\n",
    "# Evaluate on the test set\n",
    "test_results_ID_90 = trainer.evaluate(paired_test_dataset_ID_90)\n",
    "print(f'ID 90: {test_results_ID_90}')\n",
    "results['ID 90'] = test_results_ID_90\n",
    "\n",
    "test_results_OOD_change_70 = trainer.evaluate(paired_test_dataset_OOD_change_70)\n",
    "print(f'OOD change 70: {test_results_OOD_change_70}')\n",
    "results['OOD change 70'] = test_results_OOD_change_70\n",
    "\n",
    "test_results_OOD_balanced_50 = trainer.evaluate(paired_test_dataset_OOD_balanced_50)\n",
    "print(f'OOD balanced 50: {test_results_OOD_balanced_50}')\n",
    "results['OOD balanced 50'] = test_results_OOD_balanced_50\n",
    "\n",
    "test_results_OOD_change_30 = trainer.evaluate(paired_test_dataset_OOD_change_30)\n",
    "print(f'OOD change 30: {test_results_OOD_change_30}')\n",
    "results['OOD change 30'] = test_results_OOD_change_30\n",
    "\n",
    "test_results_OOD_flip_10 = trainer.evaluate(paired_test_dataset_OOD_flip_10)\n",
    "print(f'OOD flip: {test_results_OOD_flip_10}')\n",
    "results['OOD flip'] = test_results_OOD_flip_10\n",
    "\n",
    "test_results_OOD_original_0 = trainer.evaluate(paired_test_dataset_OOD_original_0)\n",
    "print(f'OOD original: {test_results_OOD_original_0}')\n",
    "results['OOD original'] = test_results_OOD_original_0\n",
    "\n",
    "# Manually save the results\n",
    "with open(f'./results/yelp/causal-sft-c/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "plot_training_metrics(trainer)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 6. Causal SFT - Phi"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Initialise Causal SFT model \"bert-base-uncased\" (unfreezed all layers) with a linear head!\n",
      "Total Parameters: 252520324\n",
      "Trainable Parameters: 143038084\n",
      "Percentage of Trainable Parameters: 56.6442%\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='630' max='630' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [630/630 03:50, Epoch 10/10]\n",
       "    </div>\n",
       "    <table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       " <tr style=\"text-align: left;\">\n",
       "      <th>Step</th>\n",
       "      <th>Training Loss</th>\n",
       "      <th>Validation Loss</th>\n",
       "      <th>F1</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <td>63</td>\n",
       "      <td>0.553600</td>\n",
       "      <td>0.480170</td>\n",
       "      <td>0.897497</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>126</td>\n",
       "      <td>0.347000</td>\n",
       "      <td>0.458397</td>\n",
       "      <td>0.896461</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>189</td>\n",
       "      <td>0.234900</td>\n",
       "      <td>0.478256</td>\n",
       "      <td>0.894985</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>252</td>\n",
       "      <td>0.136200</td>\n",
       "      <td>0.597498</td>\n",
       "      <td>0.896983</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>315</td>\n",
       "      <td>0.077100</td>\n",
       "      <td>0.557801</td>\n",
       "      <td>0.897996</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>378</td>\n",
       "      <td>0.014200</td>\n",
       "      <td>0.610454</td>\n",
       "      <td>0.897443</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>441</td>\n",
       "      <td>-0.010400</td>\n",
       "      <td>0.741708</td>\n",
       "      <td>0.900498</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>504</td>\n",
       "      <td>-0.035200</td>\n",
       "      <td>0.809426</td>\n",
       "      <td>0.898995</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>567</td>\n",
       "      <td>-0.041500</td>\n",
       "      <td>0.784480</td>\n",
       "      <td>0.897481</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <td>630</td>\n",
       "      <td>-0.046300</td>\n",
       "      <td>0.726945</td>\n",
       "      <td>0.898983</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table><p>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "\n",
       "    <div>\n",
       "      \n",
       "      <progress value='271' max='63' style='width:300px; height:20px; vertical-align: middle;'></progress>\n",
       "      [63/63 00:45]\n",
       "    </div>\n",
       "    "
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train: {'eval_loss': 0.24710184335708618, 'eval_f1': 0.9294219715784531, 'eval_runtime': 10.587, 'eval_samples_per_second': 755.647, 'eval_steps_per_second': 5.951, 'epoch': 10.0}\n",
      "validation: {'eval_loss': 0.46388599276542664, 'eval_f1': 0.896460629154236, 'eval_runtime': 2.9764, 'eval_samples_per_second': 671.948, 'eval_steps_per_second': 5.376, 'epoch': 10.0}\n",
      "ID 90: {'eval_loss': 0.4651881456375122, 'eval_f1': 0.8961938863076672, 'eval_runtime': 5.5092, 'eval_samples_per_second': 726.064, 'eval_steps_per_second': 5.809, 'epoch': 10.0}\n",
      "OOD change 70: {'eval_loss': 1.077047348022461, 'eval_f1': 0.7262139097236082, 'eval_runtime': 5.5035, 'eval_samples_per_second': 726.811, 'eval_steps_per_second': 5.814, 'epoch': 10.0}\n",
      "OOD balanced 50: {'eval_loss': 1.6872895956039429, 'eval_f1': 0.5535255784894751, 'eval_runtime': 5.4961, 'eval_samples_per_second': 727.786, 'eval_steps_per_second': 5.822, 'epoch': 10.0}\n",
      "OOD change 30: {'eval_loss': 2.293692111968994, 'eval_f1': 0.3840611268318277, 'eval_runtime': 5.4908, 'eval_samples_per_second': 728.494, 'eval_steps_per_second': 5.828, 'epoch': 10.0}\n",
      "OOD flip: {'eval_loss': 2.907759189605713, 'eval_f1': 0.2069220690993789, 'eval_runtime': 5.5017, 'eval_samples_per_second': 727.043, 'eval_steps_per_second': 5.816, 'epoch': 10.0}\n",
      "OOD original: {'eval_loss': 0.9861125946044922, 'eval_f1': 0.6649078288582091, 'eval_runtime': 5.5259, 'eval_samples_per_second': 723.866, 'eval_steps_per_second': 5.791, 'epoch': 10.0}\n"
     ]
    },
    {
     "data": {
      "image/png": "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MjNENe3LA44rGIJTabz8zZEfq977LNx2220aPXq03nzzTX300UeqU6eOLr/8cknSxIkT9eqrr2rSpElq1aqVAgICNHbsWOXn55/RuYt7TrLtpGmEf/75p4YOHaqnn35avXv3VkhIiKZPn66JEyee1ecwDOOUcxf3nsemyZ24zXmO84VP954nrn/qqad0ww036IcfftCPP/6oJ598UtOnT9eAAQN0++23q3fv3vrhhx80b948TZgwQRMnTtS99957TvUAACqJg9vNsLRmqpSXYa7zDpTaDJM6jpRqNrG2PlRM3v5Sq+vMJWu/tPFrczrfnpXSP/PNxTvQfOhx68FSve4E7wqA4FQam+2Mp8tZbfDgwbrvvvv0+eef6+OPP9bIkSNd3/QvXrxY/fr104033ijJvGdp27Ztatas2Rmdu3nz5kpISNDevXsVFWVOXfzjjz+K7PP777+rTp06euyxx1zrdu/eXWQfb29vORyOUt/r448/VnZ2tmvU6ffff5fdblfjxhdmDvCxz5eYmOgaddq0aZPS09OL/B41btxYjRs31v33369hw4bpo48+0oABAyRJMTExGjVqlEaNGqXx48frvffeIzgBgDtyOs1RgmXvmL8eU72Bee9S22GS76n38ALFCqxpPrOr053mc6HWfWGGqMO7zVHMtdOkwAgzZLUeIkW04iG7FiE4uZHAwEANGTJEjz76qNLT0zVixAjXtoYNG2rmzJlaunSpqlWrpldeeUUpKSlnHJx69uypJk2a6Oabb9bEiROVkZFRJCAde4+EhARNnz5dHTp00A8//KBZs2YV2adu3brauXOn1qxZo+joaAUFBcnHp2hbzuHDh+vJJ5/ULbfcoqeeekr79+/Xvffeq5tuusk1Te9cORyOU54h5e3trZ49e6p169YaPny4Jk2a5GoO0b17d7Vv315HjhzRgw8+qOuuu0716tVTUlKSli9frkGDBkmSxo4dqz59+qhx48Y6dOiQfvnllzP+vQUAVBJHDktrPpeWvyel7Ti60ma2mO50h1T/MslOw2Kch7BG0mWPSZc+KiX+ZQaoDV9LWSnSH2+YS63m5ihUq+ulkOjSz4kyw99uN3Pbbbfp0KFD6tmzp2JjY13rH3/8ccXFxal3797q0aOHIiIi1L9//zM+r91u16xZs5SXl6eOHTvq9ttv13PPPVdkn379+un+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Kjg42OpycIFxvasWrnfVwvWuerjmVQvXu2SGYSgzM1NRUVGy20u+i6nKjTjZ7XZFR0dbXYbbCA4O5i9hFcL1rlq43lUL17vq4ZpXLVzv0yttpOkYmkMAAAAAQCkITgAAAABQCoITzomPj4+efPJJ+fj4WF0KygHXu2rhelctXO+qh2tetXC9y06Vaw4BAAAAAGeLEScAAAAAKAXBCQAAAABKQXACAAAAgFIQnAAAAACgFAQnuPz222+65pprFBUVJZvNpm+++abIdsMw9NRTTykqKkp+fn7q0aOHNm7cWGSfvLw83XvvvQoLC1NAQICuvfZaJSUlleOnwJmaMGGCOnTooKCgINWqVUv9+/fX1q1bi+zDNXcfkydPVuvWrV0PQOzcubN+/PFH13autXubMGGCbDabxo4d61rHNXcfTz31lGw2W5ElIiLCtZ1r7Z727NmjG2+8UTVq1JC/v7/atm2rlStXurZz3csewQku2dnZatOmjd54441it7/00kt65ZVX9MYbb2j58uWKiIjQFVdcoczMTNc+Y8eO1axZszR9+nQtWbJEWVlZuvrqq+VwOMrrY+AMLVq0SPfcc4/+/PNPzZ8/X4WFherVq5eys7Nd+3DN3Ud0dLReeOEFrVixQitWrNBll12mfv36uf4T5Vq7r+XLl+vdd99V69ati6znmruXFi1aKDk52bWsX7/etY1r7X4OHTqkrl27ysvLSz/++KM2bdqkiRMnKjQ01LUP1/0CMIBiSDJmzZrleu10Oo2IiAjjhRdecK3Lzc01QkJCjLffftswDMM4fPiw4eXlZUyfPt21z549ewy73W789NNP5VY7zk1qaqohyVi0aJFhGFzzqqBatWrG+++/z7V2Y5mZmUajRo2M+fPnG927dzfuu+8+wzD4++1unnzySaNNmzbFbuNau6eHH37YuPjii0+7net+YTDihDOyc+dOpaSkqFevXq51Pj4+6t69u5YuXSpJWrlypQoKCorsExUVpZYtW7r2QcWVnp4uSapevbokrrk7czgcmj59urKzs9W5c2eutRu755571LdvX/Xs2bPIeq65+9m2bZuioqJUr149DR06VDt27JDEtXZXs2fPVvv27XX99derVq1aateund577z3Xdq77hUFwwhlJSUmRJIWHhxdZHx4e7tqWkpIib29vVatW7bT7oGIyDEPjxo3TxRdfrJYtW0rimruj9evXKzAwUD4+Pho1apRmzZql5s2bc63d1PTp07Vq1SpNmDDhlG1cc/fSqVMnffLJJ5o7d67ee+89paSkqEuXLjp48CDX2k3t2LFDkydPVqNGjTR37lyNGjVKY8aM0SeffCKJv+MXiqfVBaBysdlsRV4bhnHKupOdyT6w1ujRo7Vu3TotWbLklG1cc/fRpEkTrVmzRocPH9bMmTN1yy23aNGiRa7tXGv3kZiYqPvuu0/z5s2Tr6/vaffjmruHPn36uL5u1aqVOnfurAYNGujjjz/WRRddJIlr7W6cTqfat2+v559/XpLUrl07bdy4UZMnT9bNN9/s2o/rXrYYccIZOdad5+SfQKSmprp+mhEREaH8/HwdOnTotPug4rn33ns1e/Zs/frrr4qOjnat55q7H29vbzVs2FDt27fXhAkT1KZNG/3vf//jWruhlStXKjU1VfHx8fL09JSnp6cWLVqk1157TZ6enq5rxjV3TwEBAWrVqpW2bdvG3283FRkZqebNmxdZ16xZMyUkJEji//ALheCEM1KvXj1FRERo/vz5rnX5+flatGiRunTpIkmKj4+Xl5dXkX2Sk5O1YcMG1z6oOAzD0OjRo/X111/rl19+Ub169Yps55q7P8MwlJeXx7V2Q5dffrnWr1+vNWvWuJb27dtr+PDhWrNmjerXr881d2N5eXnavHmzIiMj+fvtprp27XrKI0T+/vtv1alTRxL/h18wFjSkQAWVmZlprF692li9erUhyXjllVeM1atXG7t37zYMwzBeeOEFIyQkxPj666+N9evXG8OGDTMiIyONjIwM1zlGjRplREdHGwsWLDBWrVplXHbZZUabNm2MwsJCqz4WTuOuu+4yQkJCjIULFxrJycmuJScnx7UP19x9jB8/3vjtt9+MnTt3GuvWrTMeffRRw263G/PmzTMMg2tdFZzYVc8wuObu5IEHHjAWLlxo7Nixw/jzzz+Nq6++2ggKCjJ27dplGAbX2h0tW7bM8PT0NJ577jlj27ZtxtSpUw1/f3/js88+c+3DdS97BCe4/Prrr4akU5ZbbrnFMAyzteWTTz5pREREGD4+PsYll1xirF+/vsg5jhw5YowePdqoXr264efnZ1x99dVGQkKCBZ8GpSnuWksyPvroI9c+XHP3ceuttxp16tQxvL29jZo1axqXX365KzQZBte6Kjg5OHHN3ceQIUOMyMhIw8vLy4iKijIGDhxobNy40bWda+2evvvuO6Nly5aGj4+P0bRpU+Pdd98tsp3rXvZshmEY1ox1AQAAAEDlwD1OAAAAAFAKghMAAAAAlILgBAAAAAClIDgBAAAAQCkITgAAAABQCoITAAAAAJSC4AQAAAAApSA4AQAAAEApCE4AgEonNTVVd955p2JjY+Xj46OIiAj17t1bf/zxhyTJZrPpm2++sbZIAIBb8bS6AAAAztagQYNUUFCgjz/+WPXr19e+ffv0888/Ky0tzerSAABuihEnAEClcvjwYS1ZskQvvviiLr30UtWpU0cdO3bU+PHj1bdvX9WtW1eSNGDAANlsNtdrSfruu+8UHx8vX19f1a9fX08//bQKCwtd2202myZPnqw+ffrIz89P9erV05dffunanp+fr9GjRysyMlK+vr6qW7euJkyYUF4fHQBgIYITAKBSCQwMVGBgoL755hvl5eWdsn358uWSpI8++kjJycmu13PnztWNN96oMWPGaNOmTXrnnXc0ZcoUPffcc0WOf/zxxzVo0CCtXbtWN954o4YNG6bNmzdLkl577TXNnj1bX3zxhbZu3arPPvusSDADALgvm2EYhtVFAABwNmbOnKmRI0fqyJEjiouLU/fu3TV06FC1bt1akjlyNGvWLPXv3991zCWXXKI+ffpo/PjxrnWfffaZHnroIe3du9d13KhRozR58mTXPhdddJHi4uL01ltvacyYMdq4caMWLFggm81WPh8WAFAhMOIEAKh0Bg0apL1792r27Nnq3bu3Fi5cqLi4OE2ZMuW0x6xcuVLPPPOMa8QqMDBQI0eOVHJysnJyclz7de7cuchxnTt3do04jRgxQmvWrFGTJk00ZswYzZs374J8PgBAxUNwAgBUSr6+vrriiiv0xBNPaOnSpRoxYoSefPLJ0+7vdDr19NNPa82aNa5l/fr12rZtm3x9fUt8r2OjS3Fxcdq5c6f+85//6MiRIxo8eLCuu+66Mv1cAICKieAEAHALzZs3V3Z2tiTJy8tLDoejyPa4uDht3bpVDRs2PGWx24//d/jnn38WOe7PP/9U06ZNXa+Dg4M1ZMgQvffee5oxY4ZmzpxJNz8AqAJoRw4AqFQOHjyo66+/Xrfeeqtat26toKAgrVixQi+99JL69esnSapbt65+/vlnde3aVT4+PqpWrZqeeOIJXX311YqJidH1118vu92udevWaf369Xr22Wdd5//yyy/Vvn17XXzxxZo6daqWLVumDz74QJL06quvKjIyUm3btpXdbteXX36piIgIhYaGWvFbAQAoRwQnAEClEhgYqE6dOunVV1/V9u3bVVBQoJiYGI0cOVKPPvqoJGnixIkaN26c3nvvPdWuXVu7du1S79699f333+uZZ57RSy+9JC8vLzVt2lS33357kfM//fTTmj59uu6++25FRERo6tSpat68ueu9X3zxRW3btk0eHh7q0KGD5syZU2TECgDgnuiqBwDAUcV14wMAQOIeJwAAAAAoFcEJAAAAAErBPU4AABzF7HUAwOkw4gQAAAAApSA4AQAAAEApCE4AAAAAUAqCEwAAAACUguAEAAAAAKUgOAEAAABAKQhOAAAAAFAKghMAAAAAlILgBAAAAACl+H8644eLAPPPzgAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "class CausalSft_Model_Phi(nn.Module):\n",
    "    def __init__(self, base_model, num_labels, max_words):\n",
    "        super().__init__()\n",
    "\n",
    "        self.max_words = max_words\n",
    "        self.base_model = copy.deepcopy(base_model)\n",
    "        self.sft_model = copy.deepcopy(base_model)\n",
    "        self.sft_classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "        self.r_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 2, self.base_model.config.hidden_size // 2),\n",
    "        )\n",
    "        self.c_2_c = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size // 4, self.base_model.config.hidden_size // 4),\n",
    "        )\n",
    "        self.num_patches = 10\n",
    "        self.patches_size = self.max_words // self.num_patches\n",
    "        self.patch_extractor = nn.Sequential(\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches, self.base_model.config.hidden_size * self.num_patches // 2),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size * self.num_patches // 2, self.base_model.config.hidden_size),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size),\n",
    "        )\n",
    "        # self.c_patch_mixer = nn.Sequential(\n",
    "        #     nn.Linear(self.base_model.config.hidden_size, self.base_model.config.hidden_size // 8),\n",
    "        #     nn.ReLU(),\n",
    "        #     nn.Linear(self.base_model.config.hidden_size // 8, self.base_model.config.hidden_size // 8),\n",
    "        # )\n",
    "        self.classifier = nn.Linear(self.base_model.config.hidden_size, num_labels)\n",
    "\n",
    "        self.cross_entropy_loss = nn.CrossEntropyLoss()\n",
    "        self.mse_loss = nn.MSELoss()\n",
    "\n",
    "        for param in self.base_model.parameters():\n",
    "            param.requires_grad = False\n",
    "\n",
    "        print(f'Initialise Causal SFT model \"{model_name}\" (unfreezed all layers) with a linear head!')\n",
    "        count_parameters(self)\n",
    "\n",
    "    def entropy_maximization(self, feature):\n",
    "        p = F.softmax(feature, dim=-1)\n",
    "        log_p = F.log_softmax(feature, dim=-1)\n",
    "        entropy = -torch.mean(torch.mean(p * log_p, dim=-1))  # Maximize entropy\n",
    "        return -entropy  # Minimize negative entropy\n",
    "\n",
    "    def forward(self, input_ids1, attention_mask1, input_ids2, attention_mask2, input_ids3, attention_mask3, input_ids4, attention_mask4, labels=None):\n",
    "        B, _ = input_ids1.size()\n",
    "        # avoid unwanted computational graph\n",
    "        outputs_sft2 = self.sft_model(input_ids2, attention_mask=attention_mask2)[1]\n",
    "        outputs_sft2_logits = self.sft_classifier(outputs_sft2)\n",
    "        # pull out R0 and R1 using another data point\n",
    "        outputs_base3 = self.base_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        outputs_sft3 = self.sft_model(input_ids3, attention_mask=attention_mask3)[1]\n",
    "        # optimise C from R0 and R1\n",
    "        base_C3= self.r_2_c(outputs_base3)[:, :self.base_model.config.hidden_size//4]\n",
    "        sft_C3 = self.r_2_c(outputs_sft3)[:, :self.base_model.config.hidden_size//4]\n",
    "        # gather C \n",
    "        outputs_sft = self.sft_model(input_ids1, attention_mask=attention_mask1)[1]\n",
    "        sft_C = self.r_2_c(outputs_sft)[:, :self.base_model.config.hidden_size//4]\n",
    "        # extract non-contextual embeddings\n",
    "        with torch.no_grad():\n",
    "            embeddings_1 = self.sft_model.embeddings(input_ids1, attention_mask1)[:, :self.max_words]\n",
    "            embeddings_2 = self.sft_model.embeddings(input_ids2, attention_mask2)[:, :self.max_words]\n",
    "            embeddings_4 = self.sft_model.embeddings(input_ids4, attention_mask4)[:, :self.max_words]\n",
    "\n",
    "        input = embeddings_1\n",
    "        patches = torch.sum(input.view(B, self.num_patches, self.patches_size, -1), dim=2)\n",
    "        Ai_samples = self.patch_extractor(patches.view(B, -1))\n",
    "        # Ai_samples = Ai_samples[torch.randperm(B).to(input_ids1.device), :]\n",
    "        Ai_Z1 = Ai_samples\n",
    "        logits = self.classifier(Ai_Z1)   \n",
    "\n",
    "        loss = None\n",
    "        if labels is not None:\n",
    "            loss_sft = self.cross_entropy_loss(outputs_sft2_logits, labels) \n",
    "            loss_identify = self.mse_loss(base_C3, sft_C3) + self.entropy_maximization(sft_C3) + self.entropy_maximization(base_C3)\n",
    "            loss_cls = self.cross_entropy_loss(logits, labels) \n",
    "            \n",
    "            loss = loss_sft + loss_identify + loss_cls \n",
    "        \n",
    "        return (loss, logits) if loss is not None else logits\n",
    "    \n",
    "set_seed(data_seed)\n",
    "model = CausalSft_Model_Phi(base_model, num_cls, max_words)\n",
    "\n",
    "# Define training arguments and trainer\n",
    "training_args = TrainingArguments(\n",
    "    overwrite_output_dir=True,\n",
    "    output_dir=f'./results/yelp/causal-sft-phi/{N}-shot-{data_seed}',\n",
    "    eval_strategy=\"steps\",\n",
    "    save_strategy=\"steps\",\n",
    "    logging_strategy=\"steps\",\n",
    "    logging_steps=0.1,\n",
    "    save_steps=0.1,\n",
    "    learning_rate=5e-5,\n",
    "    per_device_train_batch_size=128,\n",
    "    per_device_eval_batch_size=128,\n",
    "    num_train_epochs=10,\n",
    "    seed=data_seed,\n",
    "    load_best_model_at_end=True,\n",
    "    metric_for_best_model=\"eval_loss\",  # Choose model based on validation loss\n",
    "    greater_is_better=False,  # Lower validation loss is better\n",
    "    save_total_limit=1,\n",
    ")\n",
    "\n",
    "\n",
    "trainer = Trainer(\n",
    "    model=model,\n",
    "    args=training_args,\n",
    "    train_dataset=paired_train_dataset,\n",
    "    eval_dataset=paired_validation_dataset,\n",
    "    tokenizer=tokenizer,\n",
    "    data_collator=paired_data_collator,\n",
    "    compute_metrics=compute_metrics,\n",
    ")\n",
    "\n",
    "# Train the model\n",
    "trainer.train()\n",
    "\n",
    "results = {}\n",
    "train_results = trainer.evaluate(paired_train_dataset)\n",
    "print(f'train: {train_results}')\n",
    "results['train'] = train_results\n",
    "\n",
    "valid_results = trainer.evaluate(paired_validation_dataset)\n",
    "print(f'validation: {valid_results}')\n",
    "results['valid'] = valid_results\n",
    "\n",
    "# Evaluate on the test set\n",
    "test_results_ID_90 = trainer.evaluate(paired_test_dataset_ID_90)\n",
    "print(f'ID 90: {test_results_ID_90}')\n",
    "results['ID 90'] = test_results_ID_90\n",
    "\n",
    "test_results_OOD_change_70 = trainer.evaluate(paired_test_dataset_OOD_change_70)\n",
    "print(f'OOD change 70: {test_results_OOD_change_70}')\n",
    "results['OOD change 70'] = test_results_OOD_change_70\n",
    "\n",
    "test_results_OOD_balanced_50 = trainer.evaluate(paired_test_dataset_OOD_balanced_50)\n",
    "print(f'OOD balanced 50: {test_results_OOD_balanced_50}')\n",
    "results['OOD balanced 50'] = test_results_OOD_balanced_50\n",
    "\n",
    "test_results_OOD_change_30 = trainer.evaluate(paired_test_dataset_OOD_change_30)\n",
    "print(f'OOD change 30: {test_results_OOD_change_30}')\n",
    "results['OOD change 30'] = test_results_OOD_change_30\n",
    "\n",
    "test_results_OOD_flip_10 = trainer.evaluate(paired_test_dataset_OOD_flip_10)\n",
    "print(f'OOD flip: {test_results_OOD_flip_10}')\n",
    "results['OOD flip'] = test_results_OOD_flip_10\n",
    "\n",
    "test_results_OOD_original_0 = trainer.evaluate(paired_test_dataset_OOD_original_0)\n",
    "print(f'OOD original: {test_results_OOD_original_0}')\n",
    "results['OOD original'] = test_results_OOD_original_0\n",
    "\n",
    "# Manually save the results\n",
    "with open(f'./results/yelp/causal-sft-phi/{N}-shot-{data_seed}/test_results.json', 'w') as f:\n",
    "    json.dump(results, f, indent=4)\n",
    "\n",
    "plot_training_metrics(trainer)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "robustNLP",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
