{
  "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": "lLuksYsodCSr"
      },
      "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=seed)\n",
        "    df_men_sample = df_men.sample(n=min_size, random_state=seed)\n",
        "\n",
        "    # Combine balanced dataset\n",
        "    df = pd.concat([df_women_sample, df_men_sample]).sample(frac=1, random_state=seed).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": [
        "from opacus.optimizers import AdaClipDPOptimizer, DPOptimizer\n",
        "from opacus.optimizers.optimizer import (\n",
        "    _check_processed_flag,\n",
        "    _generate_noise,\n",
        "    _mark_as_processed\n",
        ")\n",
        "\n",
        "class FairOptimizer(AdaClipDPOptimizer):\n",
        "    def __init__(self, *args, **kwargs):\n",
        "        super().__init__(*args,**kwargs)\n",
        "\n",
        "\n",
        "    def clip_and_accumulate(self):\n",
        "        per_param_norms = [\n",
        "            g.view(len(g), -1).norm(2, dim=-1) for g in self.grad_samples\n",
        "        ]\n",
        "        per_sample_norms = torch.stack(per_param_norms, dim=1).norm(2, dim=1)\n",
        "        per_sample_clip_factor = torch.tanh(self.max_grad_norm / (per_sample_norms + 1e-6))  #the only change to make it fair with tanh\n",
        "\n",
        "        # the two lines below are the only changes\n",
        "        # relative to the parent DPOptimizer class.\n",
        "        self.sample_size += len(per_sample_clip_factor)\n",
        "        self.unclipped_num += (\n",
        "            len(per_sample_clip_factor) - (per_sample_clip_factor < 1).sum()\n",
        "        )\n",
        "\n",
        "        for p in self.params:\n",
        "            _check_processed_flag(p.grad_sample)\n",
        "            grad_sample = self._get_flat_grad_sample(p)\n",
        "            grad = torch.einsum(\"i,i...\", per_sample_clip_factor, grad_sample)\n",
        "\n",
        "            if p.summed_grad is not None:\n",
        "                p.summed_grad += grad\n",
        "            else:\n",
        "                p.summed_grad = grad\n",
        "\n",
        "            _mark_as_processed(p.grad_sample)"
      ],
      "metadata": {
        "id": "zdNstZ80dDc1"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "from opacus import PrivacyEngine\n",
        "import torch.optim as optim\n",
        "from typing import List, Union\n",
        "from opacus.optimizers import DPOptimizer\n",
        "\n",
        "class fairPrivacyEngine(PrivacyEngine):\n",
        "    def __init__(self, *args, **kwargs):\n",
        "        super().__init__(*args, **kwargs)\n",
        "\n",
        "    def _prepare_optimizer(\n",
        "        self,\n",
        "        *,\n",
        "        optimizer: optim.Optimizer,\n",
        "        noise_multiplier: float,\n",
        "        max_grad_norm: Union[float, List[float]],\n",
        "        expected_batch_size: int,\n",
        "        loss_reduction: str = \"mean\",\n",
        "        distributed: bool = False,\n",
        "        clipping: str = \"flat\",\n",
        "        noise_generator=None,\n",
        "        grad_sample_mode=\"hooks\",\n",
        "        **kwargs,\n",
        "    ) -> DPOptimizer:\n",
        "        if isinstance(optimizer, DPOptimizer):\n",
        "            optimizer = optimizer.original_optimizer\n",
        "\n",
        "        generator = None\n",
        "        if self.secure_mode:\n",
        "            generator = self.secure_rng\n",
        "        elif noise_generator is not None:\n",
        "            generator = noise_generator\n",
        "\n",
        "\n",
        "        new_fair = FairOptimizer(\n",
        "            optimizer=optimizer,\n",
        "            noise_multiplier=noise_multiplier,\n",
        "            max_grad_norm=max_grad_norm,\n",
        "            expected_batch_size=expected_batch_size,\n",
        "            loss_reduction=loss_reduction,\n",
        "            generator=generator,\n",
        "            secure_mode=self.secure_mode,\n",
        "            **kwargs,\n",
        "        )\n",
        "\n",
        "        return new_fair\n",
        ""
      ],
      "metadata": {
        "id": "qj5wRIXzdE_9"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "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 random\n",
        "\n",
        "# ====================================================================================================\n",
        "# User-configurable settings\n",
        "# ====================================================================================================\n",
        "\n",
        "# Random seed (match with training seed for checkpoint compatibility)\n",
        "seed = 2025\n",
        "\n",
        "# Paths\n",
        "train_path = '/datasets/adults_extracted/adult.data'\n",
        "test_path = '/datasets/adults_extracted/adult.test'\n",
        "CHECKPOINT_PATH = \"/Checkpoints/IncomeComplex/SEED2025-16CheckpointsIncomeFairAdaptiveEqualNewComplexModel/checkpoint_epoch_16.pt\"  #Choose the correct path for the checkpoint\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",
        "seed = 2025\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",
        "epsilon = 8\n",
        "max_grad_norm = 0.1\n",
        "epochs = 16\n",
        "batch_size = 128\n",
        "lr = 0.0003\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",
        "    # Check if both classes are present for AUC computation\n",
        "    if len(np.unique(labels)) == 2 and probs is not None:\n",
        "        metrics[\"auc_roc\"] = roc_auc_score(labels, probs)\n",
        "    else:\n",
        "        metrics[\"auc_roc\"] = None  # Set AUC to None if not both classes are present\n",
        "\n",
        "    return metrics\n",
        "\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",
        "\n",
        "X_train,y_train = prepcosessNew(df_train)\n",
        "X_test,y_test = prepcosessNew(df_test)\n",
        "\n",
        "#X = X.values if hasattr(X, 'values') else np.array(X)\n",
        "#y = y.values if hasattr(y, 'values') else np.array(y)\n",
        "\n",
        "#X_train,X_test,y_train,y_test = train_test_split(X,y,test_size =0.3,random_state = seed,stratify=y)\n",
        "#X_train, X_val, y_train, y_val = train_test_split(X_train, y_train, test_size=0.1, random_state=seed)\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",
        "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",
        "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.norm1 = nn.GroupNorm(32, 128)  # Replaces BatchNorm1d\n",
        "        self.fc2 = nn.Linear(128, 64)\n",
        "        self.norm2 = nn.GroupNorm(32, 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.norm1(self.fc1(x)))\n",
        "        x = self.relu(self.norm2(self.fc2(x)))\n",
        "        x = self.relu(self.fc3(x))\n",
        "        x = self.fc4(x)\n",
        "        return x\n",
        "\n",
        "\n",
        "def load_checkpoint(checkpoint_path,X_train,train_loader):\n",
        "\n",
        "    epsilon = 8\n",
        "    max_grad_norm = 0.1\n",
        "    epochs = 16\n",
        "    batch_size = 128\n",
        "    lr = 0.0003\n",
        "\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)\n",
        "\n",
        "    base_optimizer = optim.Adam(model.parameters(), lr=lr, weight_decay=0.000005)\n",
        "\n",
        "    model.train()\n",
        "\n",
        "    sample_rate = batch_size / len(X_train)\n",
        "\n",
        "    sigma = get_noise_multiplier(\n",
        "        target_epsilon=epsilon,\n",
        "        target_delta=1e-5,\n",
        "        sample_rate=sample_rate,\n",
        "        epochs=epochs\n",
        "    )\n",
        "    print(f\"Using Noise Multiplier (sigma): {sigma:.4f}\")\n",
        "\n",
        "    m = batch_size  # users per round\n",
        "    sigma_b = m / 20        # from the paper\n",
        "\n",
        "    privacy_engine = fairPrivacyEngine()\n",
        "\n",
        "    model, optimizer, train_loader = privacy_engine.make_private(\n",
        "        module=model,\n",
        "        optimizer=base_optimizer,\n",
        "        clipping = 'adaptive',\n",
        "        data_loader=train_loader,\n",
        "        noise_multiplier=sigma,\n",
        "        max_grad_norm=max_grad_norm,\n",
        "        poisson_sampling=False, # poisson_sampling=True, If I was using drop_last = False in dataloader\n",
        "        loss_reduction=\"sum\", #sum\n",
        "        target_unclipped_quantile=0.5, # gamma = median\n",
        "        clipbound_learning_rate=0.2, # ηC\n",
        "        max_clipbound=2.0,  # upper limit on clipping norm\n",
        "        min_clipbound=0.1,  # lower limit\n",
        "        unclipped_num_std=sigma_b\n",
        "    )\n",
        "\n",
        "\n",
        "    checkpoint = torch.load(checkpoint_path)\n",
        "    optimizer.load_state_dict(checkpoint['optimizer_state_dict'])\n",
        "    #new_state_dict = {k.replace('_module.', ''): v for k, v in checkpoint['model_state_dict'].items()}\n",
        "    model.load_state_dict(checkpoint['model_state_dict'])\n",
        "\n",
        "    start_epoch = checkpoint['epoch']\n",
        "\n",
        "    print(f\"Loaded checkpoint from {checkpoint_path}, starting from epoch {start_epoch}\")\n",
        "    return model, optimizer , start_epoch\n",
        "\n",
        "\n",
        "# Initialize model\n",
        "# Load a checkpoint, Define paths\n",
        "checkpoint_path = CHECKPOINT_PATH\n",
        "\n",
        "model, optimizer, start_epoch = load_checkpoint(checkpoint_path,X_train,train_loader)\n",
        "\n",
        "micro_batch_size = 20\n",
        "grad_norms_before_clipping = []\n",
        "grad_norms_after_clipping = []\n",
        "grad_norms_after_clipping_noisy = []\n",
        "all_noises_grad_sample = []\n",
        "all_noises_grad = []\n",
        "\n",
        "\n",
        "def compute_gradient_norms(model, dataloader, optimizer, group, groupname):\n",
        "\n",
        "    model.train()  # Enable gradient tracking\n",
        "    total_grad_norm_before = 0.0\n",
        "    total_grad_norm_after = 0.0\n",
        "    count = 0\n",
        "    total_loss = 0\n",
        "    total_micro_batches = 0\n",
        "    micro_batch_size = 20\n",
        "    grad_list = []  # Initialize grad_list\n",
        "    all_logical_labels = []\n",
        "    all_logical_predictions = []\n",
        "    layer_wise_grad_samples = {}\n",
        "    cosine_similarities = []\n",
        "    angles = []\n",
        "    all_preds = []\n",
        "    all_labels = []\n",
        "    all_probs = []\n",
        "    accuracy_White = []\n",
        "    accuracy_nonWhite = []\n",
        "\n",
        "    grad_before_White = []\n",
        "    grad_before_nonWhite = []\n",
        "\n",
        "    grad_after_White = []\n",
        "    grad_after_nonWhite = []\n",
        "\n",
        "    f1score_White = []\n",
        "    f1score_nonWhite = []\n",
        "\n",
        "    ratio_White = []\n",
        "    ratio_nonWhite = []\n",
        "\n",
        "    loss_white = []\n",
        "    loss_nonWhite = []\n",
        "\n",
        "    device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
        "    class_weights = torch.tensor([1.0, 2.0]).to(device)\n",
        "    criterion = nn.CrossEntropyLoss(reduction=\"sum\", weight=class_weights)\n",
        "\n",
        "    #with torch.no_grad():\n",
        "    with BatchMemoryManager(\n",
        "            data_loader=dataloader,\n",
        "            max_physical_batch_size=micro_batch_size,\n",
        "            optimizer=optimizer\n",
        "        ) as memory_safe_data_loader:\n",
        "\n",
        "        for batch in dataloader:\n",
        "\n",
        "            total_micro_batches += 1\n",
        "\n",
        "            X_batch, y_batch = batch\n",
        "            X_batch, y_batch = X_batch.to(device), y_batch.to(device)\n",
        "\n",
        "            # Forward pass\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",
        "            total_loss += loss.item()\n",
        "            print_loss = loss.item()\n",
        "\n",
        "            # Compute and log metrics every batch\n",
        "            all_preds.extend(predicted.detach().cpu().numpy())\n",
        "            all_labels.extend(y_batch.detach().cpu().numpy())\n",
        "            #all_probs.extend(probs.detach().cpu().numpy())\n",
        "            all_probs.extend(probs[:, 1].detach().cpu().numpy())\n",
        "\n",
        "            # Backward pass\n",
        "            loss.backward()\n",
        "\n",
        "            # Compute gradient norm before clipping\n",
        "            for name, p in model.named_parameters():\n",
        "                if p.requires_grad:\n",
        "                    with torch.no_grad():\n",
        "                        grad_sample = p.grad_sample.sum(dim = 0).clone().detach()\n",
        "                        if name not in layer_wise_grad_samples:\n",
        "                            layer_wise_grad_samples[name] = torch.zeros_like(grad_sample)  # Store per-layer name\n",
        "\n",
        "                        layer_wise_grad_samples[name] += grad_sample  # Accumulate across physical batches\n",
        "\n",
        "            is_final_step = optimizer.pre_step()\n",
        "\n",
        "            if is_final_step:\n",
        "                total_grad_norm_squared = 0.0\n",
        "                total_grad_norm_squared_summed_noisy = 0.0\n",
        "                total_noise_grad = 0.0\n",
        "                all_layer_grads_before=[]\n",
        "                all_layer_grads_after=[]\n",
        "\n",
        "                for name, grad_list in layer_wise_grad_samples.items():\n",
        "                    all_layer_grads_before.append(grad_list)  # Store per-layer gradients\n",
        "                    ## Compute the L2 norm squared\n",
        "                    total_grad_norm_squared += torch.norm(grad_list, p=2) ** 2\n",
        "\n",
        "\n",
        "                overall_grad_norm = total_grad_norm_squared ** 0.5\n",
        "\n",
        "                for name, p in model.named_parameters():\n",
        "                    if p.requires_grad:\n",
        "                        with torch.no_grad():\n",
        "                            all_layer_grads_after.append(p.summed_grad.clone().detach())\n",
        "\n",
        "                            layer_grad_norm_summed_noisy = torch.norm(p.summed_grad, p=2).item()\n",
        "                            total_grad_norm_squared_summed_noisy += layer_grad_norm_summed_noisy ** 2\n",
        "\n",
        "                            grad = torch.norm(p.grad, p=2).item()\n",
        "                            total_noise_grad += (grad - layer_grad_norm_summed_noisy) ** 2\n",
        "\n",
        "\n",
        "                # Convert lists to single tensors\n",
        "                total_grad_before = torch.cat([v.view(-1) for v in all_layer_grads_before])\n",
        "                total_grad_after = torch.cat([v.view(-1) for v in all_layer_grads_after])  # Aggregate all layers\n",
        "\n",
        "\n",
        "                # Compute Cosine Similarity\n",
        "                cosine_similarity = torch.nn.functional.cosine_similarity(\n",
        "                    total_grad_before.view(1, -1),\n",
        "                    total_grad_after.view(1, -1),\n",
        "                    dim=1\n",
        "                ).item()\n",
        "\n",
        "                # Compute Angle in Degrees\n",
        "                angle = torch.acos(torch.clamp(torch.tensor(cosine_similarity), -1.0, 1.0)).item() * (180 / torch.pi)\n",
        "\n",
        "                # Store values for logging\n",
        "                cosine_similarities.append(cosine_similarity)\n",
        "                angles.append(angle)\n",
        "\n",
        "                # Compute overall gradient norm\n",
        "                overall_grad_norm = total_grad_norm_squared ** 0.5\n",
        "                overall_grad_norm_summed_noisy = total_grad_norm_squared_summed_noisy ** 0.5\n",
        "                overal_noise_grad = total_noise_grad ** 0.5\n",
        "\n",
        "                # Compute metrics after all physical batches in the logical batch are processed\n",
        "                eval_metrics = compute_metrics(np.array(all_preds), np.array(all_labels), np.array(all_probs))\n",
        "                avg_train_loss = total_loss / total_micro_batches\n",
        "\n",
        "                avg_eval_loss = total_loss / total_micro_batches\n",
        "\n",
        "                # Compute the average similarity and angle across all layers\n",
        "                avg_cosine_similarity = sum(cosine_similarities) / len(cosine_similarities)\n",
        "                avg_angle = sum(angles) / len(angles)\n",
        "\n",
        "\n",
        "                accuracy_White.append(eval_metrics['accuracy'])\n",
        "                f1score_White.append(eval_metrics['f1'])\n",
        "                grad_before_White.append(overall_grad_norm)\n",
        "                grad_after_White.append(overall_grad_norm_summed_noisy)\n",
        "                ratio_White.append(overall_grad_norm_summed_noisy/overall_grad_norm)\n",
        "                loss_white.append(avg_eval_loss)\n",
        "\n",
        "\n",
        "                all_logical_labels = []\n",
        "                all_logical_predictions = []\n",
        "                grad_list = []  # Reset grad_list\n",
        "                layer_wise_grad_samples = {}\n",
        "                all_labels = []\n",
        "                all_preds = []\n",
        "                all_probs = []\n",
        "\n",
        "\n",
        "            optimizer.zero_grad()\n",
        "            count += 1\n",
        "\n",
        "        avg_grad_norm_before_White = sum(grad_before_White)/len(grad_before_White)\n",
        "        avg_grad_norm_after_White = sum(grad_after_White) / len(grad_after_White)\n",
        "        avg_accuracy_White = sum(accuracy_White) / len(accuracy_White)\n",
        "        avg_f1_White = sum(f1score_White) / len(f1score_White)\n",
        "        avg_loss_white=sum(loss_white)/len(loss_white)\n",
        "\n",
        "        print(f\"Average Gradient Norm Before Clipping {group}: {avg_grad_norm_before_White}\")\n",
        "        print(f\"Average Gradient Norm After Clipping {group}: {avg_grad_norm_after_White}\")\n",
        "        print(f\"Average Accuracy for {group}: {avg_accuracy_White}\")\n",
        "        print(f\"Average f1score for {group}: {avg_f1_White}\")\n",
        "        print(f\"Average loss for {group}: {avg_loss_white}\")\n",
        "\n",
        "        print(f\"{avg_accuracy_White:.4f}, {avg_f1_White:.4f}, {avg_loss_white:.4f}, {avg_grad_norm_before_White:.4f}, {avg_grad_norm_after_White:.4f}\")\n",
        "\n",
        "\n",
        "\n",
        "\n",
        "# Compute gradient norms for different groups\n",
        "\n",
        "\n",
        "print(\"\\nComputing gradient norms for age below median dataset:\")\n",
        "compute_gradient_norms(model, age_below_med_loader, optimizer,'below age median' , groupname='age')\n",
        "\n",
        "print(\"\\nComputing gradient norms for age above/equal to median dataset:\")\n",
        "compute_gradient_norms(model, age_above_med_loader, optimizer,'above age median' , groupname='age')\n",
        "\n",
        "print(\"\\nComputing gradient norms for male dataset:\")\n",
        "compute_gradient_norms(model, male_loader, optimizer,'male' , groupname='gender')\n",
        "\n",
        "print(\"\\nComputing gradient norms for female dataset:\")\n",
        "compute_gradient_norms(model, female_loader, optimizer,'female' , groupname='gender')"
      ],
      "metadata": {
        "id": "VrPFCP1FdF8O"
      },
      "execution_count": null,
      "outputs": []
    }
  ]
}