{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "provenance": []
    },
    "kernelspec": {
      "name": "python3",
      "display_name": "Python 3"
    },
    "language_info": {
      "name": "python"
    }
  },
  "cells": [
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "wezpufrBTNvv"
      },
      "outputs": [],
      "source": [
        "def prepcosessNew(df):\n",
        "    columns = [\"age\", \"workclass\", \"fnlwgt\", \"education\", \"education-num\",\n",
        "           \"marital-status\", \"occupation\", \"relationship\", \"race\", \"sex\",\n",
        "           \"capital-gain\", \"capital-loss\", \"hours-per-week\", \"native-country\", \"income\"]\n",
        "\n",
        "\n",
        "    df.replace('?',np.nan,inplace = True)\n",
        "    percent_missing = df.isnull().sum()*100/len(df)\n",
        "    print(\"Number of observation before dropping Nulls:\",df.shape[0])\n",
        "    df.dropna(how='any',inplace = True)\n",
        "    print(\"Number of observation after dropping Nulls:\",df.shape[0])\n",
        "    df= df.drop_duplicates()\n",
        "    print(\"Number of observation after dropping duplicates:\",df.shape[0])\n",
        "\n",
        "\n",
        "    def income(salary):\n",
        "        if salary == '<=50K' or salary == '<=50K.':\n",
        "            return 0\n",
        "        else:\n",
        "            return 1\n",
        "\n",
        "    df['income']= df['income'].apply(income)\n",
        "\n",
        "    def hours_per_week(hours):\n",
        "        if hours< 40:\n",
        "            return 0\n",
        "        elif 40 <= hours <= 60:\n",
        "            return 1\n",
        "        else:\n",
        "            return 2\n",
        "\n",
        "    # Remove extra invisible characters and whitespace\n",
        "    df['age'] = df['age'].astype(str).str.strip()\n",
        "    # Remove non-numeric characters if any slipped through (optional)\n",
        "    df['age'] = df['age'].str.extract('(\\d+)', expand=False)\n",
        "    # Now convert to numeric\n",
        "    df['age'] = pd.to_numeric(df['age'], errors='coerce')\n",
        "    # Check how many were not convertible (optional)\n",
        "    print(\"Non-convertible values:\", df['age'].isna().sum())\n",
        "    # Drop only truly bad rows (should be very few if any)\n",
        "    df.dropna(subset=['age'], inplace=True)\n",
        "    df['age'] = df['age'].astype(int)\n",
        "\n",
        "\n",
        "    df['hours-per-week']=df['hours-per-week'].apply(hours_per_week)\n",
        "\n",
        "    hs_grad = ['HS-grad','11th','10th','9th','12th']\n",
        "    elementary = ['1st-4th','5th-6th','7th-8th']\n",
        "\n",
        "    # replace elements in list.\n",
        "    df['education'].replace(to_replace = hs_grad,value = 'HS-grad',inplace = True)\n",
        "    df['education'].replace(to_replace = elementary,value = 'elementary_school',inplace = True)\n",
        "\n",
        "    low_edu = ['Preschool','elementary_school','HS-grad','Some-college','Assoc-voc','Assoc-acdm']\n",
        "    high_edu = ['Bachelors','Masters','Doctorate','Prof-school']\n",
        "\n",
        "    #Replace elements in list\n",
        "    df['education'].replace(to_replace = low_edu,value = 0,inplace = True) #low\n",
        "    df['education'].replace(to_replace = high_edu,value = 1,inplace = True) #high\n",
        "\n",
        "    married= ['Married-spouse-absent','Married-civ-spouse','Married-AF-spouse']\n",
        "    separated = ['Separated','Divorced']\n",
        "\n",
        "    #replace elements in list.\n",
        "    df['marital-status'].replace(to_replace = married ,value = 'Married',inplace = True)\n",
        "    df['marital-status'].replace(to_replace = separated,value = 'Separated',inplace = True)\n",
        "\n",
        "    non_married=['Never-married','Separated','Widowed']\n",
        "    df['marital-status'].replace(to_replace = non_married,value = 'other',inplace = True)\n",
        "\n",
        "\n",
        "    self_employed = ['Self-emp-not-inc','Self-emp-inc']\n",
        "    govt_employees = ['Local-gov','State-gov','Federal-gov']\n",
        "\n",
        "    #replace elements in list.\n",
        "    df['workclass'].replace(to_replace = self_employed ,value = 'Self_employed',inplace = True)\n",
        "    df['workclass'].replace(to_replace = govt_employees,value = 'Govt_employees',inplace = True)\n",
        "\n",
        "    non_private = ['Govt_employees','Self_employed','Without-pay']\n",
        "    df['workclass'].replace(to_replace = non_private,value = 'non_private',inplace = True)\n",
        "\n",
        "\n",
        "    # coutry to us and non_us\n",
        "    us_country = ['United-States']\n",
        "    non_us_country = ['Holand-Netherlands','Scotland','Honduras','Hungary','Outlying-US(Guam-USVI-etc)','Yugoslavia','Laos','Thailand',\n",
        "                     'Trinadad&Tobago','Cambodia','Hong','Ireland','Ecuador','France','Greece','Peru','Nicaragua','Portugal','Taiwan',\n",
        "                     'Haiti','Iran','Columbia','Poland','Japan','Guatemala','Vietnam','Dominican-Republic','Italy','China','South','Jamaica',\n",
        "                     'England','Cuba','India','El-Salvador','Canada','Puerto-Rico','Germany','Philippines','Mexico']\n",
        "\n",
        "    df['native-country'].replace(to_replace = us_country,value = 'US',inplace = True)\n",
        "    df['native-country'].replace(to_replace = non_us_country,value = 'non_US',inplace = True)\n",
        "\n",
        "    #Race to white and non_white\n",
        "    white=['White']\n",
        "    non_white=['Black','Asian-Pac-Islander','Amer-Indian-Eskimo','Other']\n",
        "\n",
        "    df['race'].replace(to_replace = white,value = 'white',inplace = True)\n",
        "    df['race'].replace(to_replace = non_white,value = 'non_white',inplace = True)\n",
        "\n",
        "    #Relationship to married and other\n",
        "    rel_married=['Husband','Wife']\n",
        "    rel_other=['Not-in-family','Own-child','Unmarried','Other-relative']\n",
        "\n",
        "    df['relationship'].replace(to_replace=rel_married, value='married',inplace=True)\n",
        "    df['relationship'].replace(to_replace=rel_other, value='other',inplace=True)\n",
        "\n",
        "    #occupation to office/heacy_work,other\n",
        "    office=['Adm-clerical','Exec-managerial','Prof-specialty','Sales','Tech-support']\n",
        "    heavy_work=['Craft-repair','Machine-op-inspct','Transport-moving','Handlers-cleaners','Farming-fishing','Priv-house-serv','Armed-Forces']\n",
        "    other=['Other-service','Protective-serv']\n",
        "\n",
        "    df['occupation'].replace(to_replace=office, value='office',inplace=True)\n",
        "    df['occupation'].replace(to_replace=heavy_work, value='heavy_work',inplace=True)\n",
        "    df['occupation'].replace(to_replace=other, value='other',inplace=True)\n",
        "\n",
        "    # Convert 'sex' column to binary if it's in string format\n",
        "    df['sex'] = df['sex'].map({'Male': 1, 'Female': 0})\n",
        "\n",
        "    '''#Undersample based on gender\n",
        "    df_women = df[df['sex'] == 0]\n",
        "    df_men = df[df['sex'] == 1]\n",
        "\n",
        "    min_size = min(len(df_women), len(df_men))\n",
        "    df_women_sample = df_women.sample(n=min_size, random_state=42)\n",
        "    df_men_sample = df_men.sample(n=min_size, random_state=42)\n",
        "\n",
        "    # Combine balanced dataset\n",
        "    df = pd.concat([df_women_sample, df_men_sample]).sample(frac=1, random_state=42).reset_index(drop=True)'''\n",
        "    # Separate the data by 'sex' and 'income'\n",
        "    df_women_class_0 = df[(df['sex'] == 0) & (df['income'] == 0)]  # Women with income <=50K\n",
        "    df_women_class_1 = df[(df['sex'] == 0) & (df['income'] == 1)]  # Women with income >50K\n",
        "    df_men_class_0 = df[(df['sex'] == 1) & (df['income'] == 0)]  # Men with income <=50K\n",
        "    df_men_class_1 = df[(df['sex'] == 1) & (df['income'] == 1)]  # Men with income >50K\n",
        "\n",
        "    # Find the minimum size of the minority class (income 1) in the women and men groups\n",
        "    min_size_women = len(df_women_class_0) + len(df_women_class_1)  # All women\n",
        "    min_size_men = len(df_men_class_1)  # All men with income 1\n",
        "\n",
        "    diff = min_size_women - min_size_men\n",
        "    # For men with income 0, sample the same number as men with income 1\n",
        "    df_men_class_0_sampled = df_men_class_0.sample(n=diff, random_state=seed)\n",
        "\n",
        "    # Combine all women with all men having income 1 and the sampled men with income 0\n",
        "    df = pd.concat([df_women_class_0, df_women_class_1, df_men_class_1, df_men_class_0_sampled])\n",
        "\n",
        "    # Shuffle the data\n",
        "    df = df.sample(frac=1, random_state=seed).reset_index(drop=True)\n",
        "\n",
        "    # Check the final size and distribution\n",
        "    print('Size after undersampling:', df.shape[0])\n",
        "    print('Women with income 0 size:', df_women_class_0.shape[0], 'Women with income 1 size:', df_women_class_1.shape[0])\n",
        "    print('Men with income 0 size:', df_men_class_0_sampled.shape[0], 'Men with income 1 size:', df_men_class_1.shape[0])\n",
        "\n",
        "    #Age\n",
        "    median_age = df['age'].median()\n",
        "    df['age'] = df['age'].apply(lambda x: 0 if x <= median_age else 1)\n",
        "\n",
        "    # Different from the paper ... this is outlier\n",
        "    print(\"Number of observation before removing:\",df.shape)\n",
        "    index_gain = df[df['capital-gain'] == 99999].index\n",
        "    df.drop(labels = index_gain,axis = 0,inplace =True)\n",
        "    print(\"Number of observation after removing:\",df.shape)\n",
        "\n",
        "\n",
        "    df['capital-gain']=df['capital-gain'].apply(lambda x: 0 if x<=5000 else 1)\n",
        "    df['capital-loss']=df['capital-loss'].apply(lambda x: 0 if x<=40 else 1)\n",
        "\n",
        "    df=df.drop(columns=['fnlwgt','education-num'])\n",
        "\n",
        "    #Separate categorical and numberical columns\n",
        "    cat_col = df.dtypes[df.dtypes == 'object']\n",
        "    num_col = df.dtypes[df.dtypes != 'object']\n",
        "\n",
        "    adult_df =df.copy()\n",
        "\n",
        "    #Mapping\n",
        "    adult_df['workclass'] = adult_df['workclass'].map({'Private': 1, 'non_private': 0})\n",
        "    adult_df['marital-status'] = adult_df['marital-status'].map({'Married': 1, 'other': 0})\n",
        "    adult_df['relationship'] = adult_df['relationship'].map({'married': 1, 'other': 0})\n",
        "    adult_df['race'] = adult_df['race'].map({'white': 1, 'non_white': 0})\n",
        "    adult_df['native-country'] = adult_df['native-country'].map({'US': 1, 'non_US': 0})\n",
        "    #adult_df['sex'] = adult_df['sex'].map({'Male': 1, 'Female': 0})\n",
        "    adult_df = pd.get_dummies(adult_df, columns=['occupation'])\n",
        "\n",
        "\n",
        "    y = adult_df['income']\n",
        "    adult_df.drop(labels = ['income'],axis = 1,inplace = True)\n",
        "    X = adult_df\n",
        "    return X,y"
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "import numpy as np\n",
        "import pandas as pd\n",
        "from sklearn.pipeline import Pipeline\n",
        "from sklearn.base import TransformerMixin\n",
        "from sklearn.preprocessing import MinMaxScaler,StandardScaler\n",
        "from sklearn.model_selection import train_test_split,cross_val_score,GridSearchCV\n",
        "import torch\n",
        "import torch.nn as nn\n",
        "import torch.optim as optim\n",
        "from torch.utils.data import Dataset, DataLoader\n",
        "from sklearn.preprocessing import StandardScaler\n",
        "import opacus\n",
        "from opacus.accountants.utils import get_noise_multiplier\n",
        "from opacus.utils.batch_memory_manager import BatchMemoryManager\n",
        "from sklearn.metrics import accuracy_score, precision_score, recall_score, f1_score, roc_auc_score\n",
        "from sklearn.preprocessing import LabelEncoder\n",
        "import torch.nn.functional as F\n",
        "import optuna\n",
        "from transformers import get_linear_schedule_with_warmup\n",
        "from torch.utils.data import ConcatDataset\n",
        "import sys\n",
        "from tqdm.auto import tqdm\n",
        "import os\n",
        "import wandb\n",
        "from sklearn.utils.class_weight import compute_class_weight\n",
        "\n",
        "\n",
        "# ====================================================================================================\n",
        "# User-configurable settings\n",
        "# ====================================================================================================\n",
        "\n",
        "# Random seed\n",
        "SEED = 2025\n",
        "\n",
        "# Paths\n",
        "train_path = '/datasets/adults_extracted/adult.data'\n",
        "test_path = '/datasets/adults_extracted/adult.test'\n",
        "\n",
        "\n",
        "# Set seeds for reproducibility\n",
        "seed = 42\n",
        "np.random.seed(seed)\n",
        "torch.manual_seed(seed)\n",
        "torch.cuda.manual_seed(seed)\n",
        "torch.backends.cudnn.deterministic = True\n",
        "torch.backends.cudnn.benchmark = False\n",
        "\n",
        "g = torch.Generator()\n",
        "g.manual_seed(seed)\n",
        "\n",
        "wandb.init(\n",
        "    project=\"income-nonDP_newmodel\",  # Change project name if needed\n",
        "    name=\"income-nonDP-genderEqual_newmodel\",  # Give a unique run name\n",
        ")\n",
        "\n",
        "columns = [\"age\", \"workclass\", \"fnlwgt\", \"education\", \"education-num\",\n",
        "       \"marital-status\", \"occupation\", \"relationship\", \"race\", \"sex\",\n",
        "       \"capital-gain\", \"capital-loss\", \"hours-per-week\", \"native-country\", \"income\"]\n",
        "\n",
        "def test_model(model, dataloader):\n",
        "    model.eval()\n",
        "    total_loss = 0.0\n",
        "    all_preds = []\n",
        "    all_labels = []\n",
        "    all_probs = []\n",
        "\n",
        "    with torch.no_grad():\n",
        "        for X_batch, y_batch in dataloader:\n",
        "            X_batch = X_batch.to(device)\n",
        "            y_batch = y_batch.to(device)\n",
        "\n",
        "            outputs = model(X_batch)\n",
        "            probs = F.softmax(outputs, dim=1)\n",
        "            preds = torch.argmax(probs, dim=1)\n",
        "\n",
        "            loss = criterion(outputs, y_batch)\n",
        "            total_loss += loss.item()\n",
        "\n",
        "            all_preds.extend(preds.cpu().numpy())\n",
        "            all_labels.extend(y_batch.cpu().numpy())\n",
        "            all_probs.extend(probs[:, 1].cpu().numpy())\n",
        "\n",
        "    avg_loss = total_loss / len(dataloader)\n",
        "    metrics = compute_metrics(all_preds, all_labels, all_probs)\n",
        "\n",
        "    print(\"Test Results:\")\n",
        "    print(f\"Avg Loss: {avg_loss:.4f} | Accuracy: {metrics['accuracy']:.4f} | Precision: {metrics['precision']:.4f} | Recall: {metrics['recall']:.4f} | F1: {metrics['f1']:.4f} | AUROC: {metrics['auc_roc']:.4f}\")\n",
        "    return metrics\n",
        "\n",
        "class EarlyStopping:\n",
        "    def __init__(self, patience=3, threshold=0.0):\n",
        "        self.patience = patience\n",
        "        self.threshold = threshold\n",
        "        self.counter = 0\n",
        "        self.best_score = None\n",
        "        self.early_stop = False\n",
        "\n",
        "    def __call__(self, val_score):\n",
        "        if self.best_score is None:\n",
        "            self.best_score = val_score\n",
        "        elif val_score < self.best_score + self.threshold:\n",
        "            self.counter += 1\n",
        "            if self.counter >= self.patience:\n",
        "                self.early_stop = True\n",
        "        else:\n",
        "            self.best_score = val_score\n",
        "            self.counter = 0\n",
        "        return self.early_stop\n",
        "\n",
        "df_train = pd.read_csv(train_path, names=columns, skipinitialspace=True)\n",
        "df_test = pd.read_csv(test_path, names=columns, skipinitialspace=True)\n",
        "\n",
        "X_train , y_train = prepcosessNew(df_train)\n",
        "X_test , y_test = prepcosessNew(df_test)\n",
        "\n",
        "\n",
        "#df = pd.concat([df_train, df_test], ignore_index=True)\n",
        "\n",
        "early_stopper = EarlyStopping(patience=5, threshold=0.0)\n",
        "\n",
        "epochs = 8 #60\n",
        "batch_size = 32\n",
        "\n",
        "\n",
        "def compute_metrics(preds, labels, probs=None):\n",
        "    preds = np.array(preds)\n",
        "    labels = np.array(labels)\n",
        "    metrics = {\n",
        "        \"accuracy\": accuracy_score(labels, preds),\n",
        "        \"precision\": precision_score(labels, preds, zero_division=0),\n",
        "        \"recall\": recall_score(labels, preds, zero_division=0),\n",
        "        \"f1\": f1_score(labels, preds, zero_division=0),\n",
        "    }\n",
        "    if probs is not None:\n",
        "        metrics[\"auc_roc\"] = roc_auc_score(labels, probs)\n",
        "    return metrics\n",
        "\n",
        "\n",
        "\n",
        "#X = pd.read_csv(\"X_income.csv\")\n",
        "#y = pd.read_csv(\"Y_income.csv\")\n",
        "\n",
        "#X_train,X_val,y_train,y_val = train_test_split(X_train,y_train,test_size =0.15,random_state = 42, stratify=y_train)\n",
        "\n",
        "X_test, X_val, y_test, y_val = train_test_split(X_test, y_test, test_size=0.5, random_state=seed,stratify=y_test)\n",
        "\n",
        "\n",
        "X_train = X_train.to_numpy(dtype=np.float32)  # Convert DataFrame to NumPy\n",
        "y_train = y_train.to_numpy(dtype=np.int64)  # Ensure y_train is integer\n",
        "X_test = X_test.to_numpy(dtype=np.float32)\n",
        "y_test = y_test.to_numpy(dtype=np.int64)\n",
        "X_val = X_val.to_numpy(dtype=np.float32)\n",
        "y_val = y_val.to_numpy(dtype=np.int64)\n",
        "\n",
        "\n",
        "# Convert to PyTorch tensors\n",
        "X_train_tensor = torch.tensor(X_train, dtype=torch.float32)\n",
        "y_train_tensor = torch.tensor(y_train, dtype=torch.long)\n",
        "X_test_tensor = torch.tensor(X_test, dtype=torch.float32)\n",
        "y_test_tensor = torch.tensor(y_test, dtype=torch.long)\n",
        "X_val_tensor = torch.tensor(X_val, dtype=torch.float32)\n",
        "y_val_tensor = torch.tensor(y_val, dtype=torch.long)\n",
        "\n",
        "\n",
        "age_col_idx = columns.index('age')  # Find index of 'age' in original DataFrame\n",
        "age_below_med_mask = X_test[:, age_col_idx] == 0\n",
        "X_test_below_med, y_test_below_med = X_test[age_below_med_mask], y_test[age_below_med_mask]\n",
        "X_test_above_med, y_test_above_med = X_test[~age_below_med_mask], y_test[~age_below_med_mask]\n",
        "\n",
        "\n",
        "X_test_below_med_tensor = torch.tensor(X_test_below_med, dtype=torch.float32)\n",
        "y_test_below_med_tensor = torch.tensor(y_test_below_med, dtype=torch.long)\n",
        "X_test_above_med_tensor = torch.tensor(X_test_above_med, dtype=torch.float32)\n",
        "y_test_above_med_tensor = torch.tensor(y_test_above_med, dtype=torch.long)\n",
        "\n",
        "\n",
        "gender_col_idx = columns.index('sex') # Find index of 'age' in original DataFrame\n",
        "male_mask = X_test[:, gender_col_idx] == 1\n",
        "female_mask = X_test[:, gender_col_idx] == 0\n",
        "\n",
        "X_test_male, y_test_male = X_test[male_mask], y_test[male_mask]\n",
        "X_test_female, y_test_female = X_test[female_mask], y_test[female_mask]\n",
        "\n",
        "X_test_male_tensor = torch.tensor(X_test_male, dtype=torch.float32)\n",
        "y_test_male_tensor = torch.tensor(y_test_male, dtype=torch.long)\n",
        "X_test_female_tensor = torch.tensor(X_test_female, dtype=torch.float32)\n",
        "y_test_female_tensor = torch.tensor(y_test_female, dtype=torch.long)\n",
        "\n",
        "\n",
        "\n",
        "class IncomeDataset(Dataset):\n",
        "    def __init__(self, X, y):\n",
        "        self.X = X\n",
        "        self.y = y\n",
        "\n",
        "    def __len__(self):\n",
        "        return len(self.X)\n",
        "\n",
        "    def __getitem__(self, idx):\n",
        "        return self.X[idx], self.y[idx]\n",
        "\n",
        "# Create dataset objects\n",
        "train_dataset = IncomeDataset(X_train_tensor, y_train_tensor)\n",
        "test_dataset = IncomeDataset(X_test_tensor, y_test_tensor)\n",
        "val_dataset =  IncomeDataset(X_val_tensor, y_val_tensor)\n",
        "male_dataset = IncomeDataset(X_test_male_tensor,y_test_male_tensor)\n",
        "female_dataset = IncomeDataset(X_test_female_tensor,y_test_female_tensor)\n",
        "age_above_med = IncomeDataset(X_test_above_med_tensor,y_test_above_med_tensor)\n",
        "age_below_med = IncomeDataset(X_test_below_med_tensor,y_test_below_med_tensor)\n",
        "\n",
        "# Create data loaders\n",
        "train_loader = DataLoader(train_dataset, batch_size=batch_size, shuffle=True,\n",
        "                          worker_init_fn=lambda worker_id: torch.manual_seed(seed + worker_id),  # Fixes worker shuffling\n",
        "                          generator=g)  # Ensures deterministic shuffling\n",
        "test_loader = DataLoader(test_dataset, batch_size=batch_size, shuffle=False)\n",
        "male_loader = DataLoader(male_dataset, batch_size=batch_size, shuffle=True)\n",
        "female_loader = DataLoader(female_dataset, batch_size=batch_size, shuffle=True)\n",
        "age_above_med_loader = DataLoader(age_above_med, batch_size=batch_size, shuffle=True)\n",
        "age_below_med_loader = DataLoader(age_below_med, batch_size=batch_size, shuffle=True)\n",
        "val_loader = DataLoader(val_dataset, batch_size=batch_size, shuffle=False)\n",
        "\n",
        "\n",
        "class IncomeModel(nn.Module):\n",
        "    def __init__(self, input_size):\n",
        "        super(IncomeModel, self).__init__()\n",
        "        self.fc1 = nn.Linear(input_size, 128)\n",
        "        self.bn1 = nn.BatchNorm1d(128)\n",
        "        self.fc2 = nn.Linear(128, 64)\n",
        "        self.bn2 = nn.BatchNorm1d(64)\n",
        "        self.fc3 = nn.Linear(64, 32)\n",
        "        self.fc4 = nn.Linear(32, 2)\n",
        "        self.relu = nn.ReLU()\n",
        "        self.dropout = nn.Dropout(0.3)\n",
        "\n",
        "    def forward(self, x):\n",
        "        x = self.relu(self.bn1(self.fc1(x)))\n",
        "        x = self.relu(self.bn2(self.fc2(x)))\n",
        "        x = self.relu(self.fc3(x))\n",
        "        x = self.fc4(x)\n",
        "        return x\n",
        "\n",
        "\n",
        "# Initialize model\n",
        "input_size = X_train.shape[1]  # Number of features\n",
        "model = IncomeModel(input_size)\n",
        "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
        "model.to(device)  # Move model to GPU if available\n",
        "\n",
        "class_weights = torch.tensor([1.0, 2.0]).to(device)\n",
        "criterion = nn.CrossEntropyLoss(reduction=\"sum\", weight=class_weights)\n",
        "optimizer = optim.Adam(model.parameters(), lr=0.00003, weight_decay = 0.000005)   #0.0003\n",
        "\n",
        "# Compute warmup steps dynamically\n",
        "warmup_ratio = 0.1  # 10% of total steps\n",
        "num_training_steps = len(train_loader) * epochs\n",
        "num_warmup_steps = int(warmup_ratio * num_training_steps)\n",
        "scheduler = get_linear_schedule_with_warmup(optimizer, num_warmup_steps=num_warmup_steps, num_training_steps=num_training_steps)\n",
        "\n",
        "best_val_f1 = 0\n",
        "patience_counter = 0\n",
        "patience = 5\n",
        "#Training loop\n",
        "model.train()  # Set to training mode\n",
        "for epoch in range(epochs):\n",
        "    model.train()\n",
        "    total_loss = 0\n",
        "    all_preds, all_labels, all_probs = [], [], []\n",
        "\n",
        "    for X_batch, y_batch in train_loader:\n",
        "        X_batch, y_batch = X_batch.to(device), y_batch.to(device)\n",
        "\n",
        "        optimizer.zero_grad()\n",
        "        outputs = model(X_batch)\n",
        "        probs = F.softmax(outputs, dim=1)  # [batch_size, 2]\n",
        "        predicted = torch.argmax(probs, dim=1)  # [batch_size]\n",
        "\n",
        "        loss = criterion(outputs, y_batch)\n",
        "        loss.backward()\n",
        "        optimizer.step()\n",
        "        scheduler.step()\n",
        "\n",
        "        total_loss += loss.item()\n",
        "        all_preds.extend(predicted.cpu().numpy())\n",
        "        all_labels.extend(y_batch.cpu().numpy())\n",
        "        #all_probs.extend(probs.detach().cpu().numpy())\n",
        "        all_probs.extend(probs[:, 1].detach().cpu().numpy())\n",
        "\n",
        "    train_metrics = compute_metrics(all_preds, all_labels, all_probs)\n",
        "    print(f\"Epoch {epoch+1} | Train Loss: {total_loss:.4f} | Train F1: {train_metrics['f1']:.4f} | Train Accuracy:  {train_metrics['accuracy']:.4f}\")\n",
        "\n",
        "    # Evaluation\n",
        "    model.eval()\n",
        "    val_loss = 0\n",
        "    all_eval_preds, all_eval_labels, all_eval_probs = [], [], []\n",
        "    with torch.no_grad():\n",
        "        for X_batch, y_batch in val_loader:\n",
        "            X_batch, y_batch = X_batch.to(device), y_batch.to(device)\n",
        "            outputs = model(X_batch)\n",
        "            probs = F.softmax(outputs, dim=1)  # [batch_size, 2]\n",
        "            predicted = torch.argmax(probs, dim=1)  # [batch_size]\n",
        "            loss = criterion(outputs, y_batch)\n",
        "\n",
        "            val_loss += loss.item()\n",
        "            all_eval_preds.extend(predicted.cpu().numpy())\n",
        "            all_eval_labels.extend(y_batch.cpu().numpy())\n",
        "            #all_eval_probs.extend(probs.detach().cpu().numpy())\n",
        "            all_eval_probs.extend(probs[:, 1].detach().cpu().numpy())\n",
        "\n",
        "\n",
        "    val_metrics = compute_metrics(all_eval_preds, all_eval_labels, all_eval_probs)\n",
        "    print(f\"Validation F1: {val_metrics['f1']:.4f} | Accuracy: {val_metrics['accuracy']:.4f} |precision: {val_metrics['precision']:.4f}| Recall: {val_metrics['recall']:.4f} | Patience Counter: {early_stopper.counter}\")\n",
        "\n",
        "    wandb.log({\n",
        "        \"epoch\": epoch + 1,\n",
        "        \"train_loss\": total_loss,\n",
        "        \"train_f1\": train_metrics[\"f1\"],\n",
        "        \"val_loss\": val_loss,\n",
        "        \"val_f1\": val_metrics[\"f1\"],\n",
        "        \"val_accuracy\": val_metrics[\"accuracy\"],\n",
        "        \"train_accuracy\": train_metrics[\"accuracy\"],\n",
        "        \"train_precicion\": train_metrics[\"precision\"],\n",
        "        \"train_recall\": train_metrics[\"recall\"],\n",
        "    })\n",
        "\n",
        "    # Early stopping logic\n",
        "    # if early_stopper(val_metrics[\"accuracy\"]):\n",
        "    #     print(f\"Early stopping triggered at epoch {epoch+1}. Best Validation F1: {early_stopper.best_score:.4f}\")\n",
        "    #     break\n",
        "\n",
        "\n",
        "\n",
        "print('Final Testing phase with val data:')\n",
        "print(\"\\n=== Full Test Set Evaluation ===\")\n",
        "test_model(model, test_loader)\n",
        "print(\"\\n=== Test on Female Subset ===\")\n",
        "test_model(model, female_loader)\n",
        "print(\"\\n=== Test on Male Subset ===\")\n",
        "test_model(model, male_loader)\n",
        "print(\"\\n=== Test on age below median Subset ===\")\n",
        "test_model(model, age_below_med_loader)\n",
        "print(\"\\n=== Test on age above Subset ===\")\n",
        "test_model(model, age_above_med_loader)"
      ],
      "metadata": {
        "id": "NDnfX6BtTZ6A"
      },
      "execution_count": null,
      "outputs": []
    }
  ]
}