{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "1b53d07a-7071-43dd-a006-be552e8f0aff",
   "metadata": {},
   "outputs": [],
   "source": [
    "from collections import OrderedDict\n",
    "import sys\n",
    "import os\n",
    "\n",
    "from copy import deepcopy\n",
    "import numpy as np\n",
    "import random\n",
    "import matplotlib.pyplot as plt\n",
    "import pickle\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import torch.optim as optim\n",
    "from torchvision import datasets, transforms\n",
    "from torch.utils.data import DataLoader, Subset"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6c1acaec-eb34-45e7-b2e1-20232ecfc6a8",
   "metadata": {},
   "source": [
    "## Dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "12a29368-1bbd-4f3e-85f8-9ccc212ac060",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "########################## also check below ##########################################\n",
    "#https://github.com/YasMinSdt/FedAlt/blob/master/data/partition/sort_and_partition.py#L2????\n",
    "# Function to create a non-IID and heterogeneous dataset for each client\n",
    "def create_non_iid_datasets(num_clients, dataset, beta):\n",
    "    # Split the original dataset into non-IID subsets for each \n",
    "\n",
    "       \n",
    "    # transform = transforms.Compose([transforms.ToTensor(), transforms.Normalize((0.5, 0.5, 0.5), (0.5, 0.5, 0.5))])\n",
    "    if dataset == \"CIFAR10\":\n",
    "        transform = transforms.Compose([transforms.ToTensor(), transforms.Normalize((0.4914, 0.4822, 0.4465), (0.2023, 0.1994, 0.2010))])\n",
    "        # Load CIFAR-10 dataset\n",
    "        train_dataset = datasets.CIFAR10(root='./data', train=True, download=True, transform=transform)\n",
    "        test_dataset = datasets.CIFAR10(root='./data', train=False, download=True, transform=transform)\n",
    "        num_classes = 10\n",
    "    elif dataset == \"CIFAR100\":\n",
    "        transform = transforms.Compose([transforms.ToTensor(), transforms.Normalize((0.5071, 0.4865, 0.4409), (0.2673, 0.2564, 0.2762))])\n",
    "        # Load CIFAR-100 dataset\n",
    "        train_dataset = datasets.CIFAR100(root='./data', train=True, download=True, transform=transform)\n",
    "        test_dataset = datasets.CIFAR100(root='./data', train=False, download=True, transform=transform)\n",
    "        num_classes = 100\n",
    "    elif dataset == \"MNIST\":\n",
    "        transform = transforms.Compose([transforms.ToTensor(), transforms.Normalize((0.1307,), (0.3015,))])\n",
    "        train_dataset = datasets.MNIST(root='./data', train=True, download=True, transform=transform)\n",
    "        test_dataset = datasets.MNIST(root='./data', train=False, download=True, transform=transform)\n",
    "        num_classes = 10\n",
    "    # elif dataset == \"EMNIST\":\n",
    "    #     transform = transforms.Compose([transforms.ToTensor(), transforms.Normalize((0.1307,), (0.3015,))])\n",
    "    #     train_dataset = datasets.EMNIST(root='./data', train=True, download=True, transform=transform)\n",
    "    #     test_dataset = datasets.EMNIST(root='./data', train=False, download=True, transform=transform)\n",
    "    else:\n",
    "        raise ValueError(f'the dataset {dataset} is not supported')\n",
    "    train_loaders = []\n",
    "    test_loaders = []\n",
    "    train_loaders = shuffle_and_split(train_dataset, num_clients, num_classes, beta)\n",
    "    test_loaders = shuffle_and_split(test_dataset, num_clients, num_classes, beta)\n",
    "\n",
    "    return train_loaders, test_loaders\n",
    "\n",
    "\n",
    "\n",
    "# def shuffle_and_split(train_dataset, num_clients, num_classes):\n",
    "#     client_datasets = []\n",
    "#     dataset_size = len(train_dataset)\n",
    "\n",
    "#     # Dynamically create class counts for each class\n",
    "#     # You can adjust the distribution strategy here\n",
    "#     class_counts = [random.randint(100, 1000) for _ in range(num_classes)]\n",
    "    \n",
    "#     for i in range(num_clients):\n",
    "#         # Randomly sample classes with different counts for each client\n",
    "#         sampled_classes = torch.multinomial(torch.tensor(class_counts).float(), num_classes, replacement=True)\n",
    "\n",
    "#         # Extract indices for each class for the client\n",
    "#         indices = [idx for idx, label in enumerate(train_dataset.targets) if label in sampled_classes]\n",
    "\n",
    "#         # Shuffle the dataset and create a subset for each client\n",
    "#         random.shuffle(indices)\n",
    "#         sampled_dataset = Subset(train_dataset, indices)\n",
    "#         client_datasets.append(sampled_dataset)\n",
    "    \n",
    "#     return client_datasets\n",
    "\n",
    "\n",
    "def shuffle_and_split(train_dataset, num_clients, num_classes, beta=0.1):\n",
    "\n",
    "    # To store the indices assigned to each client.\n",
    "    idx_batch = [[] for _ in range(num_clients)]  # Initialize empty list for each client.\n",
    "\n",
    "    # Extract the labels from the dataset\n",
    "    dataset_labels = train_dataset.targets\n",
    "\n",
    "    # Loop over each class to allocate class-specific indices to clients\n",
    "    for k in range(num_classes):\n",
    "        # Find all the indices that belong to class k\n",
    "        idx_k = np.where(np.array(dataset_labels) == k)[0]\n",
    "        np.random.shuffle(idx_k)  # Shuffle the indices to ensure randomness within class k\n",
    "        \n",
    "        # Sample the proportions from a Dirichlet distribution\n",
    "        proportions = np.random.dirichlet(np.repeat(alpha, num_clients))\n",
    "        \n",
    "        # Adjust proportions to make sure clients get some data\n",
    "        proportions = np.array([p * (len(idx_j) < len(dataset_labels) / num_clients) \n",
    "                                for p, idx_j in zip(proportions, idx_batch)])\n",
    "        \n",
    "        # Normalize the proportions so they sum to 1\n",
    "        proportions = proportions / proportions.sum()\n",
    "\n",
    "        # Split the indices according to the cumulative proportions\n",
    "        split_indices = (np.cumsum(proportions) * len(idx_k)).astype(int)[:-1]\n",
    "        \n",
    "        # Assign these indices to clients based on proportions\n",
    "        idx_batch = [idx_j + idx.tolist() for idx_j, idx in zip(idx_batch, np.split(idx_k, split_indices))]\n",
    "\n",
    "    # Create the Subset objects for each client, using the indices collected for each client\n",
    "    client_datasets = [Subset(train_dataset, idxs) for idxs in idx_batch]\n",
    "    \n",
    "    return client_datasets\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "def compute_class_distribution(dataset, num_classes):\n",
    "    \"\"\"Computes the class distribution for a given dataset.\"\"\"\n",
    "    class_counts = torch.zeros(num_classes)\n",
    "    targets = [dataset.dataset.targets[idx] for idx in dataset.indices]\n",
    "    for t in targets:\n",
    "        class_counts[t] += 1\n",
    "    class_distribution = class_counts / len(targets)  # Normalize to get probabilities\n",
    "    return class_distribution.numpy()\n",
    "\n",
    "def estimate_alpha(client_datasets, num_classes):\n",
    "    \"\"\"Estimates the Dirichlet parameter alpha based on client class distributions.\"\"\"\n",
    "    # Collect all class distributions\n",
    "    class_distributions = [compute_class_distribution(dataset, num_classes) for dataset in client_datasets]\n",
    "\n",
    "    # Mean of class distributions\n",
    "    mean_distribution = np.mean(class_distributions, axis=0)\n",
    "\n",
    "    # Estimate alpha using method of moments\n",
    "    alpha = np.sum(mean_distribution) ** 2 / np.sum(mean_distribution ** 2)\n",
    "    \n",
    "    return alpha\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "dd4bd73b-89f3-429e-b47e-e3406db77589",
   "metadata": {},
   "source": [
    "## Mixup\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "1621308f-847c-43ff-b970-e7c7ced4a53f",
   "metadata": {},
   "outputs": [],
   "source": [
    "# lambda have beta distribution / 2 is just an example\n",
    "#lam = np.random(2,2)\n",
    "\n",
    "\n",
    "def get_layer_num(model):\n",
    "    count = 0\n",
    "    for layer in model.children():\n",
    "        if any(p.requires_grad for p in layer.parameters()):\n",
    "            count += 1\n",
    "        # print(f'size model is : {count} , layer {name}')\n",
    "        # for name1, param in layer.named_parameters():\n",
    "        #     print(f\"Layer Name: {count} , param: {name1}\")\n",
    "    # print(f'size model is : {count}')\n",
    "    return count\n",
    "\n",
    "############################ Beta Distribution for creating lambda ###########################\n",
    "def get_lambda(alpha=1.0):\n",
    "    if alpha > 0.:\n",
    "        lam = np.random.beta(alpha, alpha)\n",
    "    else:\n",
    "        lam = 1.\n",
    "    return lam\n",
    "\n",
    "def mixup_criterion(y_a, y_b, lam):\n",
    "    return lambda criterion, pred: lam * criterion(pred, y_a) + (1 - lam) * criterion(pred, y_b)\n",
    "\n",
    "\n",
    "def Dynamic_lambda(mu , n, i,  phase):\n",
    "    if phase == \"broadcast\" :\n",
    "        return min(1, mu * (n - i))\n",
    "    elif phase == \"aggregation\" : \n",
    "        return max(0, 1 - (i * mu))\n",
    "\n",
    "\n",
    "def mixup(a,b, lam):\n",
    "    # lamb = torch.tensor(lam).to(device)\n",
    "    # print(f'deviceeeee 1 is {a.device} and {b.device}')\n",
    "    out = a*lam + b*(1-lam)\n",
    "    #t1 = target.data.cpu().numpy()\n",
    "    #t2 = target[indices].data.cpu().numpy()\n",
    "    #print (np.sum(t1==t2))\n",
    "    return out\n",
    "\n",
    " ####################################################layer-wise function ############################################\n",
    "def layerwise_mixup(Gm, Lm , mix_factor , phase ):\n",
    "    count = 0\n",
    "    global_param = deepcopy(OrderedDict(Gm.named_parameters()))\n",
    "    local_param = deepcopy(OrderedDict(Lm.named_parameters()))\n",
    "    num = get_layer_num(Lm)\n",
    "    for (name1, layer_Gm), (name2, layer_Lm) in zip(Gm.named_children(), Lm.named_children()):\n",
    "        if name1 == name2:\n",
    "            if any(p.requires_grad for p in layer_Lm.parameters()):\n",
    "                if phase == \"broadcast\":\n",
    "                    count += 1\n",
    "                    # print (f'Mixup is applied in phase {phase} and for layer {count}!')\n",
    "                    mix_degree = Dynamic_lambda(mix_factor, num, count,  phase)\n",
    "                    layer_Lm_mixup = Mixup_global_to_local(layer_Gm.state_dict(), layer_Lm.state_dict() , mix_degree)\n",
    "                    local_param.update(layer_Lm_mixup)\n",
    "                elif phase == \"aggregation\":\n",
    "                    count += 1\n",
    "                    # print (f'Mixup is applied in phase {phase} and for layer {count}!')\n",
    "                    mix_degree = Dynamic_lambda(mix_factor, num, count,  phase)\n",
    "                    layer_Gm_mixup = Mixup_local_to_global(layer_Gm.state_dict(), layer_Lm.state_dict() , mix_degree)\n",
    "                    global_param.update(layer_Gm_mixup) # is this really going to update everything?\n",
    "        else:\n",
    "            print(f\"Layer do not match: global layer{name1} vs local layer{name2}!\")\n",
    "    if phase == \"aggregation\": \n",
    "        return global_param\n",
    "    else: \n",
    "        return local_param \n",
    "        \n",
    "\n",
    "def Mixup_global_to_local(Gm, Lm , mix_factor):\n",
    "    Lm_mixup = OrderedDict(Lm)\n",
    "    assert len(Gm) == len(Lm), \"Gm == Lm\"\n",
    "    lam = mix_factor ### should change that later\n",
    "    # print('lm_miiix')\n",
    "    for gm , (name , lm) in zip( Gm.values() , Lm.items()):\n",
    "        Lm_mixup[name] = mixup(gm.data ,lm , lam)\n",
    "    return Lm_mixup\n",
    "\n",
    "\n",
    "def Mixup_local_to_global(Gm, Lm , mix_factor):\n",
    "    Gm_mixup = OrderedDict(Gm)\n",
    "    lam = mix_factor ### should change that later\n",
    "    # print('gm_miiix')\n",
    "    for (name , gm) , lm in zip( Gm.items() , Lm.values()):\n",
    "        Gm_mixup[name] = mixup(gm, lm.data , lam)\n",
    "    return Gm_mixup"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e17c5c5b-a575-4916-9682-15d6c877df23",
   "metadata": {},
   "source": [
    "## Model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "577b8332-31bd-4f31-a890-ddec881cbf86",
   "metadata": {},
   "outputs": [],
   "source": [
    "class PartialModel(nn.Module):\n",
    "    def __init__(self):\n",
    "        super(PartialModel, self).__init__()\n",
    "        self.layer_list = []\n",
    "        self.global_param_name = OrderedDict()\n",
    "        self.local_param_name = OrderedDict()\n",
    "        self.local_params = self.local_param_name #??? we can use orderddict instead of list\n",
    "        self.global_params = self.global_param_name #??? we can use orderddict instead of list (need to change them everywhere)\n",
    "        self.body_params = []\n",
    "        self.head_params = []\n",
    "        \n",
    "\n",
    "\n",
    "\n",
    "    def BABU_split(self):\n",
    "        temp = OrderedDict()\n",
    "        self.body_params = [p for name, p in self.named_parameters() if 'fc' not in name]\n",
    "        self.head_params = [p for name, p in self.named_parameters() if 'fc' in name]\n",
    "        for name, param in self.named_parameters():\n",
    "            if 'fc' not in name:\n",
    "                temp[name] = param.detach().clone().data\n",
    "                self.global_param_name[name] = param\n",
    "            else:\n",
    "                self.local_param_name[name] = param\n",
    "\n",
    "    def split(self, split_layer):\n",
    "        temp = OrderedDict()\n",
    "        split_layer_list = [str(layer) for layer in range(split_layer+1 , 20)]\n",
    "\n",
    "        for name, param in self.named_parameters():\n",
    "            if any (str in name.split(\".\")[0] for str in split_layer_list):\n",
    "                # temp[name] = param.detach().clone().data\n",
    "                self.local_param_name[name] = param  \n",
    "                \n",
    "            else:\n",
    "                self.global_param_name[name] = param\n",
    "                temp[name] = param.detach().clone().data\n",
    "        for name in temp:\n",
    "            print (f'global params are {name}')\n",
    "\n",
    "\n",
    "        \n",
    "        \n",
    "    \n",
    "\n",
    "# # Define the layer number you're interested in (e.g., layer k)\n",
    "# k = 1\n",
    "\n",
    "# # Access parameters of layer k\n",
    "# layer_k_params = list(model.children())[k].parameters()\n",
    "\n",
    "# Print layer k parameters\n",
    "# for param in layer_k_params:\n",
    "#     print(param)\n",
    "\n",
    "\n",
    "# class SimpleCNN_CIFAR100(nn.Module):\n",
    "#     def __init__(self, num_classes=100):\n",
    "#         super(SimpleCNN, self).__init__()\n",
    "        \n",
    "#         # First Conv Block: 3x32x32 -> 32x32x32 (Conv) -> 32x16x16 (MaxPool)\n",
    "#         self.conv1 = nn.Conv2d(3,32,3)\n",
    "#         self.bn1 = nn.BatchNorm2d(32)\n",
    "#         self.pool = nn.MaxPool2d(kernel_size=2, stride=2)\n",
    "\n",
    "#         # Second Conv Block: 32x16x16 -> 64x16x16 (Conv) -> 64x8x8 (MaxPool)\n",
    "#         self.conv2 = nn.Conv2d(32, 64, 3)\n",
    "#         self.bn2 = nn.BatchNorm2d(64)\n",
    "        \n",
    "\n",
    "#         # Third Conv Block: 64x8x8 -> 128x8x8 (Conv) -> 128x4x4 (MaxPool)\n",
    "#         self.conv3 = nn.Conv2d(64, 128, 3)\n",
    "#         self.bn3 = nn.BatchNorm2d(128)\n",
    "\n",
    "#         # Fully Connected Layers\n",
    "#         self.fc4 = nn.Linear(128 * 4 * 4, 512)  # Flatten 128 feature maps of size 4x4\n",
    "#         self.bn4 = nn.BatchNorm1d(512)\n",
    "#         self.fc5 = nn.Linear(512, 256)\n",
    "#         self.bn5 = nn.BatchNorm1d(256)\n",
    "#         self.fc6 = nn.Linear(256, num_classes)\n",
    "\n",
    "#         # Dropout to reduce overfitting\n",
    "#         self.dropout = nn.Dropout(0.5)\n",
    "#         self.flatten = nn.Flatten()\n",
    "\n",
    "#     def forward(self, x):\n",
    "#         # Apply Conv -> BatchNorm -> ReLU -> MaxPool sequence\n",
    "#         x = self.pool(F.relu(self.bn1(self.conv1(x))))\n",
    "#         x = self.pool(F.relu(self.bn2(self.conv2(x))))\n",
    "#         x = self.pool(F.relu(self.bn3(self.conv3(x))))\n",
    "\n",
    "#         # Flatten the output for fully connected layers\n",
    "#         # x = x.view(-1, 128 * 4 * 4)\n",
    "#         self.flatten(x)\n",
    "\n",
    "#         # Fully connected layers with BatchNorm and Dropout\n",
    "        # x = F.relu(self.bn4(self.fc4(x)))\n",
    "        # x = self.dropout(x)\n",
    "        # x = F.relu(self.bn5(self.fc5(x)))\n",
    "        # x = self.dropout(x)\n",
    "        # x = self.fc6(x)\n",
    "        \n",
    "        # return x\n",
    "\n",
    "# class SimpleCNN_CIFAR100(PartialModel):\n",
    "#     def __init__(self):\n",
    "#         super(SimpleCNN_CIFAR100, self).__init__()\n",
    "  \n",
    "#         # input: 3 x 32 x 32\n",
    "#         self.conv1 = nn.Conv2d(3, 64, kernel_size=5)\n",
    "#         self.conv2 = nn.Conv2d(64, 64, kernel_size=5)\n",
    "        \n",
    "\n",
    "#         self.activation = nn.ReLU()\n",
    "#         self.flatten = nn.Flatten()\n",
    "#         self.pool = nn.MaxPool2d(2, 2) # output: 64 x 16 x 16\n",
    "        \n",
    "#         self.fc3 = nn.Linear(64*5*5, 384)\n",
    "#         self.fc4 = nn.Linear(384, 192)\n",
    "#         self.fc5 = nn.Linear(192, 100)\n",
    "        \n",
    "#     def forward(self, x):\n",
    "#         x = self.conv1(x)\n",
    "#         x = self.pool(x)\n",
    "#         x = self.activation(x)\n",
    "        \n",
    "#         x = self.conv2(x)\n",
    "#         x = self.pool(x)\n",
    "#         x = self.activation(x)\n",
    "        \n",
    "#         x = torch.reshape(x, (x.shape[0], -1))\n",
    "\n",
    "#         x = self.fc3(x)\n",
    "#         x = self.activation(x)\n",
    "\n",
    "#         x = self.fc4(x)\n",
    "#         x = self.activation(x)\n",
    "\n",
    "#         x = self.fc5(x)\n",
    "#         return x\n",
    "\n",
    "\n",
    "# class SimpleCNN_CIFAR100(PartialModel):\n",
    "#     def __init__(self):\n",
    "#         super(SimpleCNN_CIFAR100, self).__init__()\n",
    "  \n",
    "#         # input: 3 x 32 x 32\n",
    "#         self.conv1 = nn.Conv2d(3, 32, kernel_size=3, padding=1)\n",
    "#         self.conv2 = nn.Conv2d(32, 64, kernel_size=3, stride=1, padding=1)\n",
    "#         self.conv3 = nn.Conv2d(64, 128, kernel_size=3, stride=1, padding=1)\n",
    "#         self.conv4 = nn.Conv2d(128, 128, kernel_size=3, stride=1, padding=1)\n",
    "\n",
    "#         self.conv5 = nn.Conv2d(128, 256, kernel_size=3, stride=1, padding=1)\n",
    "#         self.conv6 = nn.Conv2d(256, 256, kernel_size=3, stride=1, padding=1)\n",
    "\n",
    "    #     self.activation = nn.ReLU(True)\n",
    "    #     self.flatten = nn.Flatten()\n",
    "    #     self.pool = nn.MaxPool2d(2, 2) # output: 64 x 16 x 16\n",
    "        \n",
    "    #     self.fc7 = nn.Linear(256*4*4, 1024)\n",
    "    #     self.fc8 = nn.Linear(1024, 512)\n",
    "    #     self.fc9 = nn.Linear(512, 100)\n",
    "        \n",
    "    # def forward(self, x):\n",
    "    #     x = self.conv1(x)\n",
    "    #     x = self.activation(x)\n",
    "    #     x = self.conv2(x)\n",
    "    #     x = self.activation(x)\n",
    "    #     x = self.pool(x)\n",
    "\n",
    "        \n",
    "    #     x = self.conv3(x)\n",
    "    #     x = self.activation(x)\n",
    "    #     x = self.conv4(x)\n",
    "    #     x = self.activation(x)\n",
    "    #     x = self.pool(x)\n",
    "\n",
    "    #     x = self.conv5(x)\n",
    "    #     x = self.activation(x)\n",
    "    #     x = self.conv6(x)\n",
    "    #     x = self.activation(x)\n",
    "    #     x = self.pool(x)\n",
    "\n",
    "    #     x = self.flatten(x)\n",
    "\n",
    "    #     x = self.fc7(x)\n",
    "    #     x = self.activation(x)\n",
    "\n",
    "    #     x = self.fc8(x)\n",
    "    #     x = self.activation(x)\n",
    "\n",
    "    #     x = self.fc9(x)\n",
    "    #     return x\n",
    "\n",
    "\n",
    "class One_Block(nn.Module):\n",
    "    # from [FedBABU](https://github.com/jhoon-oh/FedBABU/blob/master/models/Nets.py)\n",
    "    def __init__(self, in_planes, out_planes, kernel_size , stride=1 , state=True ):\n",
    "        super(One_Block, self).__init__()\n",
    "        self.conv1 = nn.Conv2d(in_planes, out_planes, kernel_size=kernel_size , bias=state )\n",
    "        self.bn1 = nn.BatchNorm2d(out_planes , track_running_stats = state)\n",
    "\n",
    "    def forward(self, x):\n",
    "        out = F.relu(self.bn1(self.conv1(x)))\n",
    "        return out\n",
    "        \n",
    "\n",
    "class Two_Block(nn.Module):\n",
    "    # from [FedBABU](https://github.com/jhoon-oh/FedBABU/blob/master/models/Nets.py)\n",
    "    def __init__(self, in_planes, out_planes, stride=1):\n",
    "        super(Two_Block, self).__init__()\n",
    "        \n",
    "        self.conv1 = nn.Conv2d(in_planes, in_planes, kernel_size=3, stride=stride, padding=1, groups=in_planes, bias=False)\n",
    "        self.bn1 = nn.BatchNorm2d(in_planes, track_running_stats=False)\n",
    "        self.conv2 = nn.Conv2d(in_planes, out_planes, kernel_size=1, stride=1, padding=0, bias=False)\n",
    "        self.bn2 = nn.BatchNorm2d(out_planes, track_running_stats=False)\n",
    "\n",
    "    def forward(self, x):\n",
    "        out = F.relu(self.bn1(self.conv1(x)))\n",
    "        out = F.relu(self.bn2(self.conv2(out)))\n",
    "        return out\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "class MobileNet_Cifar(PartialModel):\n",
    "    # (128,2) means conv planes=128, conv stride=2, by default conv stride=1\n",
    "    cfg = [64, (128,2), 128, (256,2), 256, (512,2), 512, 512, 512, 512, 512, (1024,2), 1024]\n",
    "    \n",
    "    def __init__(self, num_classes):\n",
    "        super(MobileNet_Cifar, self).__init__()\n",
    "        # self.conv1 = nn.Conv2d(3, 32, kernel_size=3, stride=1, padding=1, bias=False)\n",
    "        # self.bn1 = nn.BatchNorm2d(32, track_running_stats=False)\n",
    "        self.layers1 = self._make_layer1(3, 32, 3 , state=False)\n",
    "        self.layers2 = self._make_layers(32, 64 , stride=1)\n",
    "        self.layers3 = self._make_layers(64,128, stride=2)\n",
    "        self.layers4 = self._make_layers(128, 128, stride=1)\n",
    "        self.layers5 = self._make_layers(128, 256, stride=2)\n",
    "        self.layers6 = self._make_layers(256, 256, stride=1)\n",
    "        self.layers7 = self._make_layers(256, 512 , stride=2)\n",
    "        self.layers8 = self._make_layers(512, 512 , stride=1)\n",
    "        self.layers9 = self._make_layers(512, 512 , stride=1)\n",
    "        self.layers10 = self._make_layers(512, 512 , stride=1)\n",
    "        self.layers11= self._make_layers(512, 512 , stride=1)\n",
    "        self.layers12 = self._make_layers(512, 512 , stride=1)\n",
    "        self.layers13 = self._make_layers(512, 1024 , stride=2)\n",
    "        self.layers14 = self._make_layers(1024, 1024, stride=1)\n",
    "        # self.activation = F.relu()\n",
    "        self.fc15 = nn.Linear(1024, num_classes)\n",
    "\n",
    "    def _make_layers(self, in_planes, out_planes, stride=1 ):\n",
    "        layers = []\n",
    "        layers.append(Two_Block(in_planes, out_planes, stride))\n",
    "        return nn.Sequential(*layers)\n",
    "\n",
    "\n",
    "    def _make_layer1(self, in_planes, out_planes, kernel , state ):\n",
    "        layers = []\n",
    "        layers.append(One_Block(in_planes, out_planes, kernel , state = state ))\n",
    "        return nn.Sequential(*layers)\n",
    "\n",
    "    \n",
    "    def forward(self, x):\n",
    "        x = self.layers1(x)\n",
    "        # x = F.relu(self.bn1(self.conv1(x)))\n",
    "        x = self.layers2(x)\n",
    "        x = self.layers3(x)\n",
    "        x = self.layers4(x)\n",
    "        x = self.layers5(x)\n",
    "        x = self.layers6(x)\n",
    "        x = self.layers7(x)\n",
    "        x = self.layers8(x)\n",
    "        x = self.layers9(x)\n",
    "        x = self.layers10(x)\n",
    "        x = self.layers11(x)\n",
    "        x = self.layers12(x)\n",
    "        x = self.layers13(x)\n",
    "        x = self.layers14(x)\n",
    "        x = F.avg_pool2d(x, 2)\n",
    "        x = x.view(x.size(0), -1)\n",
    "        logits = self.fc15(x)\n",
    "\n",
    "        return logits\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "# Define a simple CNN model for CiFAR10 # you can extend it furthur according to FB code with arbitary number of blocks\n",
    "class SimpleCNN_CIFAR100(PartialModel):\n",
    "    def __init__(self, num_classes):\n",
    "        super(SimpleCNN_CIFAR100, self).__init__()\n",
    "        self.layer1 = self._make_layers(3, 64, 3)\n",
    "        self.layer2 = self._make_layers(64, 64, 5)\n",
    "        self.layer3 = self._make_layers(64, 64, 5)\n",
    "        self.layer4 = self._make_layers(64, 64, 5)\n",
    "        \n",
    "        self.pool = nn.MaxPool2d(2)\n",
    "        self.activation = nn.ReLU()\n",
    "        self.flatten = nn.Flatten()\n",
    "        \n",
    "        self.fc5 = nn.Linear(64*2*2, num_classes)\n",
    "\n",
    "    def _make_layers(self, in_planes, out_planes, kernel ):\n",
    "        layers = []\n",
    "        layers.append(One_Block(in_planes, out_planes, kernel ))\n",
    "        return nn.Sequential(*layers)\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = self.layer1(x)\n",
    "        x = self.pool(x)\n",
    "\n",
    "        x = self.layer2(x)\n",
    "        x = self.pool(x)\n",
    "\n",
    "        x = self.layer3(x)\n",
    "        x = self.pool(x)\n",
    "\n",
    "        x = self.layer4(x)\n",
    "        x = self.pool(x)\n",
    "\n",
    "        x = self.flatten(x)\n",
    "\n",
    "        x = self.fc5(x)\n",
    "        return x\n",
    "        \n",
    "\n",
    "\n",
    "\n",
    "# Define a simple CNN model for CiFAR10 # you can extend it furthur according to FB code with arbitary number of blocks\n",
    "class SimpleCNN_CIFAR10(PartialModel):\n",
    "    def __init__(self, num_classes):\n",
    "        super(SimpleCNN_CIFAR10, self).__init__()\n",
    "        self.layer1 = self._make_layers(3, 6, 5)\n",
    "        self.layer2 = self._make_layers(6, 16, 5)\n",
    "        self.pool = nn.MaxPool2d(2)\n",
    "        self.activation = nn.ReLU(True)\n",
    "        self.flatten = nn.Flatten()\n",
    "        self.fc3 = nn.Linear(16 * 5 * 5, 120)\n",
    "        self.fc4 = nn.Linear(120, 84)\n",
    "        self.fc5 = nn.Linear(84, num_classes)\n",
    "\n",
    "\n",
    "    def _make_layers(self, in_planes, out_planes, kernel ):\n",
    "        layers = []\n",
    "        layers.append(One_Block(in_planes, out_planes, kernel ))\n",
    "        return nn.Sequential(*layers)\n",
    "\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = self.layer1(x)\n",
    "        x = self.pool(x)\n",
    "\n",
    "        x = self.layer2(x)\n",
    "        x = self.pool(x)\n",
    "\n",
    "        x = self.flatten(x)\n",
    "\n",
    "        x = self.fc3(x)\n",
    "        x = self.activation(x)\n",
    "\n",
    "        x = self.fc4(x)\n",
    "        x = self.activation(x)\n",
    "\n",
    "        x = self.fc5(x)\n",
    "        return x\n",
    "\n",
    "\n",
    "class SimpleCNN_MNIST(PartialModel):\n",
    "    def __init__(self, num_classes):\n",
    "        super(SimpleCNN_MNIST, self).__init__()\n",
    "        self.layer1 = self._make_layers(1, 32, 5)\n",
    "        self.layer2 = self._make_layers(32, 64, 5)\n",
    "        self.pool = nn.MaxPool2d(2)\n",
    "        self.activation = nn.ReLU(True)\n",
    "        self.flatten = nn.Flatten()\n",
    "        self.fc3 = nn.Linear(1024, 512)\n",
    "        self.fc4 = nn.Linear(512, 10)\n",
    "\n",
    "    def _make_layers(self, in_planes, out_planes, kernel ):\n",
    "        layers = []\n",
    "        layers.append(One_Block(in_planes, out_planes, kernel ))\n",
    "        return nn.Sequential(*layers)\n",
    "\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = self.layer1(x)\n",
    "        x = self.pool(x)\n",
    "\n",
    "        x = self.layer2(x)\n",
    "        x = self.pool(x)\n",
    "\n",
    "        x = self.flatten(x)\n",
    "\n",
    "        x = self.fc3(x)\n",
    "        x = self.activation(x)\n",
    "\n",
    "        x = self.fc4(x)\n",
    "        return x\n",
    "\n",
    "# GROUP_NORM_LOOKUP = {\n",
    "# 16: 2,  # -> channels per group: 8\n",
    "# 32: 4,  # -> channels per group: 8\n",
    "# 64: 8,  # -> channels per group: 8\n",
    "# 128: 8,  # -> channels per group: 16\n",
    "# 256: 16,  # -> channels per group: 16\n",
    "# 512: 32,  # -> channels per group: 16\n",
    "# 1024: 32,  # -> channels per group: 32\n",
    "# 2048: 32,  # -> channels per group: 64\n",
    "# }\n",
    "    \n",
    "    \n",
    "\n",
    "# def create_group_norm(num_channels):\n",
    "#     return nn.GroupNorm(GROUP_NORM_LOOKUP[num_channels], num_channels)\n",
    "\n",
    "\n",
    "# class EmnistResNetGN(ResNetGN):\n",
    "#     def __init__(self):\n",
    "#         super().__init__(layers=(2, 2, 2, 2), num_classes=62, original_size=False)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "# Class ResNetGN_EMNIST(PartialModel):\n",
    "#     def __init__(self):\n",
    "#         super(ResNetGN_EMNIST, self).__init__()\n",
    "#         self.inplanes = 64\n",
    "#         self.conv1 = nn.Conv2d(1, 64, 3, bias=False)\n",
    "#         self.bn1 = create_group_norm(64)\n",
    "\n",
    "#         self.layer1 = self._make_layer(64, 2)\n",
    "#         self.layer2 = self._make_layer(128, 2, stride=2)\n",
    "#         self.layer3 = self._make_layer(256, 2, stride=2)\n",
    "#         self.layer4 = self._make_layer(512, 2, stride=2)\n",
    "#         self.avgpool = nn.AdaptiveAvgPool2d((1, 1))\n",
    "#         self.fc = nn.Linear(512, 62)\n",
    "        \n",
    "#         self.pool = nn.Identity()\n",
    "#         self.activation = nn.ReLU(True)\n",
    "\n",
    "    #     for m in self.modules():\n",
    "    # if isinstance(m, nn.Conv2d):\n",
    "    #     nn.init.kaiming_normal_(m.weight, mode='fan_out', nonlinearity='relu')\n",
    "    # elif isinstance(m, (nn.BatchNorm2d, nn.GroupNorm)):\n",
    "    #     nn.init.constant_(m.weight, 1)\n",
    "    #     nn.init.constant_(m.bias, 0)\n",
    "\n",
    "\n",
    "    # def _make_layer(self, planes, blocks,\n",
    "    #                 stride: int = 1) -> nn.Sequential:\n",
    "    #     downsample = None\n",
    "    #     if stride != 1 or self.inplanes != planes:\n",
    "    #         downsample = nn.Sequential(\n",
    "    #             conv1x1(self.inplanes, planes, stride),\n",
    "    #             # nn.BatchNorm2d(planes),\n",
    "    #             create_group_norm(planes),\n",
    "    #         )\n",
    "    #     layers = []\n",
    "    #     layers.append(ResidualBlock(self.inplanes, planes, stride, downsample))\n",
    "    #     self.inplanes = planes\n",
    "    #     for _ in range(1, blocks):\n",
    "    #         layers.append(ResidualBlock(self.inplanes, planes))\n",
    "    #     return nn.Sequential(*layers)\n",
    "\n",
    "\n",
    "    # def forward(self, x: torch.Tensor) -> torch.Tensor:\n",
    "    #     x = self.conv1(x)\n",
    "    #     x = self.bn1(x)\n",
    "    #     x = self.activation(x)\n",
    "    #     x = self.pool(x)\n",
    "\n",
    "    #     x = self.layer1(x)\n",
    "    #     x = self.layer2(x)\n",
    "    #     x = self.layer3(x)\n",
    "    #     x = self.layer4(x)\n",
    "\n",
    "    #     x = self.avgpool(x)\n",
    "    #     x = torch.flatten(x, 1)\n",
    "    #     x = self.fc(self.drop_o(x))\n",
    "\n",
    "\n",
    "\n",
    "################################################### Get_num_classes ####################################################\n",
    "\n",
    "def get_model(model):    \n",
    "    if model.lower() == 'cnn':\n",
    "        MODEL = {\n",
    "            \"MNIST\": SimpleCNN_MNIST,\n",
    "            \"CIFAR10\":SimpleCNN_CIFAR10,\n",
    "            \"CIFAR100\":SimpleCNN_CIFAR10, \n",
    "            # \"EMNIST\": SimpleCNN_MNIST\n",
    "                }\n",
    "    elif model.lower() == 'mobile':\n",
    "        MODEL = {\n",
    "        \"CIFAR10\":MobileNet_Cifar,\n",
    "        \"CIFAR100\":MobileNet_Cifar\n",
    "        }\n",
    "    elif model.lower() == 'resnet18':\n",
    "        MODEL = {\n",
    "        \"CIFAR10\":ResNet18_CIFAR,\n",
    "        \"CIFAR100\":ResNet18_CIFAR\n",
    "        }  \n",
    "\n",
    "    elif model.lower() == 'resnet50':\n",
    "        MODEL = {\n",
    "        \"CIFAR10\":ResNet50_CIFAR,\n",
    "        \"CIFAR100\":ResNet50_CIFAR\n",
    "        }  \n",
    "    else:\n",
    "        exit('Error: unrecognized model')\n",
    "\n",
    "    return MODEL\n",
    "\n",
    "################################################### Get_num_classes ####################################################\n",
    "\n",
    "def get_num_classes(dataset):  \n",
    "    if dataset.lower() == 'cifar10':\n",
    "        num_classes = 10\n",
    "    elif dataset.lower() == 'cifar100':\n",
    "        num_classes = 100\n",
    "    elif dataset.lower() == 'mnist':\n",
    "        num_classes = 10\n",
    "    elif dataset.lower() == 'emnist':    \n",
    "        num_classes = 64\n",
    "\n",
    "    \n",
    "    else:\n",
    "        exit('Error: unrecognized dataset')\n",
    "\n",
    "    return num_classes\n",
    "    \n",
    "\n",
    "\n",
    "\n",
    "# model1 =  MobileNet_Cifar(10)    \n",
    "# layerwise_mixup(model1)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "93a2e887-1c08-41a7-9805-4c4271b7f9e0",
   "metadata": {},
   "outputs": [],
   "source": [
    "## Train"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "67bed70f-ca1c-4237-9465-a24253362a7d",
   "metadata": {},
   "source": [
    "## Train"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "1ec4328b-c82f-4924-a2fc-f85f11d801fd",
   "metadata": {},
   "outputs": [],
   "source": [
    "def evaluate(dataloader , model , criterion):\n",
    "    model.eval()           \n",
    "    total_loss = 0\n",
    "    num_samples = 0\n",
    "    acc = 0\n",
    "    with torch.no_grad(): \n",
    "        for x, y in dataloader:\n",
    "            x, y = x.to(device), y.to(device)\n",
    "            logit = model(x)\n",
    "            total_loss += criterion(logit, y)\n",
    "            pred = torch.softmax(logit, -1).argmax(-1)\n",
    "            acc += torch.eq(pred, y).int().sum()\n",
    "            num_samples += y.size(-1)\n",
    "    total_loss = total_loss / num_samples\n",
    "    accuracy = (acc/num_samples) * 100\n",
    "    model.train()      \n",
    "    return accuracy , total_loss   \n",
    "\n",
    "\n",
    "\n",
    "\n",
    "def train(train_loader, model , criterion, algo , optimizer ):\n",
    "    model.train().to(device)\n",
    "    for epoch in range(local_epochs):\n",
    "        loss_array = [] \n",
    "        for data, target in train_loader:\n",
    "            data , target = data.to(device), target.to(device)\n",
    "            output = model(data)\n",
    "            loss = criterion(output, target)\n",
    "            if algo == \"FedAvg\" or \"pFedMix\" or \"pMixFed\":\n",
    "                optimizer.zero_grad()\n",
    "                loss.backward()\n",
    "                optimizer.step()\n",
    "                \n",
    "            elif algo == \"FedSim\":\n",
    "                gradient = torch.autograd.grad(loss ,model.local_params.values())\n",
    "                local_model_lr = lr\n",
    "                for params, grad in zip(model.local_params.values(), gradient):\n",
    "                    params.data -= grad * local_model_lr\n",
    "                gradient = torch.autograd.grad(loss , model.global_params.values())\n",
    "                global_model_lr = lr\n",
    "                for params, grad in zip(model.global_params.values() , gradient):\n",
    "                    params.data -= grad * global_model_lr \n",
    "      \n",
    "            elif algo == \"FedAlt\": #just updating the local layers\n",
    "                # reconstructing the global layers from global model\n",
    "                gradient = torch.autograd.grad(loss ,model.local_params.values())\n",
    "                #### check this later! local_model_lr could be different\n",
    "                local_model_lr = lr\n",
    "                for params, grad in zip(model.local_params.values(), gradient):\n",
    "                    params.data -= grad * local_model_lr\n",
    "                # personalization (updating the global layers)\n",
    "                output = model(data)\n",
    "                loss = criterion(output, target)\n",
    "                gradient = torch.autograd.grad(loss , model.global_params.values())\n",
    "                global_model_lr = lr\n",
    "                for params, grad in zip(model.global_params.values() , gradient):\n",
    "                    params.data -= grad * global_model_lr \n",
    "                    \n",
    "            elif algo == \"FedBABU\": \n",
    "                loss.backward()\n",
    "                optimizer.step() \n",
    "\n",
    "            \n",
    "           \n",
    "\n",
    "\n",
    "######################################################### train local model #####################################################\n",
    "\n",
    "# Function to train a model partialy with fedRecon/FedAlt on a given dataset\n",
    "# ???? we have two LRs / personalization lr and reconstruction lr in FedRecon\n",
    "def train_model(algo, global_model, model, train_loader, test_loader, optimizer, criterion, local_epochs, lr , mix_factor):\n",
    "    #change the params from OrderdDict() --> List \n",
    "    # local_params = lambda w: list(model.local_params.values())\n",
    "    # global_params = lambda w:list(model.global_params.values())\n",
    "    \n",
    "    global_model.to(device)\n",
    "    model.to(device)\n",
    "    global_model_before = deepcopy(global_model)\n",
    "    global_params_dict = OrderedDict()\n",
    "    for name, param in model.global_params.items():\n",
    "        global_params_dict[name] = param.detach().clone().data\n",
    "\n",
    "    # for name, param in model.global_params.named_parameters():\n",
    "    #     global_params_dict[name] = param.detach().clone().data\n",
    "    if algo == \"pMixFed\":\n",
    "        Lm_mixup = layerwise_mixup(global_model, model,  mix_factor , phase='broadcast')\n",
    "        model.load_state_dict(Lm_mixup, strict=False)\n",
    "        delta_weight =  global_params_dict # or model.global_params # or global_params # **** check this --> [model.global_params.value]???\n",
    "    elif algo == \"pFedMix\":\n",
    "#????? check this with back_up_local_params in which you have to share the data of params\n",
    "        # Lm_mixup = layerwise_mixup(global_model.global_params, model.global_params,  mix_factor , phase='broadcast')\n",
    "        Lm_mixup = Mixup_global_to_local(global_model.global_params, model.global_params,  mix_factor )\n",
    "        model.load_state_dict(Lm_mixup, strict=False)\n",
    "        model.load_state_dict (model.local_params , strict=False)\n",
    "        delta_weight =  global_params_dict \n",
    "    # now it's the weight before training  \n",
    "    elif algo == \"FedAlt\" or \"FedSim\" :\n",
    "#????? check this with back_up_local_params in which you have to share the data of params\n",
    "        model.load_state_dict(global_model.global_params, strict=False)\n",
    "        model.load_state_dict (model.local_params , strict=False)\n",
    "        delta_weight =  global_params_dict # or  model.global_params or global_params # **** check this --> [model.global_params.value]???\n",
    "    elif algo == \"FedAvg\":    \n",
    "        global_model_params = deepcopy(OrderedDict(global_model.named_parameters()))\n",
    "        model.load_state_dict(global_model_params, strict=True)\n",
    "        delta_weight = deepcopy(OrderedDict(model.named_parameters()))\n",
    "    elif algo == \"FedBABU\" : \n",
    "        model.load_state_dict(global_model.global_params, strict=False)\n",
    "        model.load_state_dict(model.local_params, strict=False)\n",
    "        delta_weight = deepcopy(OrderedDict(model.named_parameters()))\n",
    "        model.zero_grad()\n",
    "    else:\n",
    "        raise ValueError(f'the algorithm {algo} is not supported')\n",
    "\n",
    "    del global_params_dict\n",
    "\n",
    "    train(train_loader, model , criterion, algo , optimizer )\n",
    "    torch.cuda.empty_cache()\n",
    "\n",
    "    avg_train_accuracy , average_train_loss = evaluate(train_loader , model , criterion)\n",
    "    avg_test_accuracy , _ = evaluate(test_loader , model , criterion)\n",
    "    torch.cuda.empty_cache()\n",
    "\n",
    "    \n",
    "\n",
    "    for param_name, param_tensor in delta_weight.items():\n",
    "        delta_weight[param_name] = param_tensor.to(device)\n",
    "        \n",
    "    with torch.no_grad(): \n",
    "        if algo == \"pFedMix\":\n",
    "            # Gm_mixup = OrderedDict(delta_weight)\n",
    "            Gm_mixup = Mixup_local_to_global(delta_weight, model.global_params , mix_factor)\n",
    "            for delta , weight_after in zip(delta_weight.values(), Gm_mixup.values()):\n",
    "                delta = weight_after\n",
    "        elif algo == \"pMixFed\":\n",
    "            # Gm_mixup = OrderedDict(delta_weight)\n",
    "            Gm_mixup = layerwise_mixup(global_model_before, model , mix_factor , phase=\"aggregation\" )\n",
    "            for delta , weight_after in zip(delta_weight.values(), Gm_mixup.values()):\n",
    "                delta = weight_after        \n",
    "        elif algo == \"FedAvg\": \n",
    "            for delta , weight_after in zip(delta_weight.values(), model.parameters()):\n",
    "                delta.sub_(weight_after)\n",
    "        elif algo == \"FedAlt\" or algo == \"FedSim\":\n",
    "            for delta , weight_after in zip(delta_weight.values(), model.global_params.values()):\n",
    "                delta.sub_(weight_after)\n",
    "        elif algo == \"FedBABU\" : \n",
    "            for delta , weight_after in zip(delta_weight.values(),  model.global_params.values()):\n",
    "                delta.add_(weight_after)\n",
    "            \n",
    "    # print (f'**** {average} ****')\n",
    "    return  average_train_loss , avg_train_accuracy , avg_test_accuracy ,  delta_weight\n",
    "\n",
    "################################################## Mixup aggregation ################################\n",
    "# def federated_mixup(delta_weights , alphas , global_lr):\n",
    "\n",
    "\n",
    "# Function to perform federated averaging\n",
    "def federated_averaging(delta_weights , alphas , global_lr):\n",
    "    with torch.no_grad():\n",
    "        # calculate w_i = w_i + delta_w_i * alpha_i \n",
    "        # normalize the alphas which is just based on num_samples for each client  \n",
    "        ################################ can change it to (measurmenet of diversity or ..)  *************************************\n",
    "        sum_alphas = sum(alphas)\n",
    "        alphas = list(map(lambda w: w / sum_alphas, alphas))\n",
    "        for delta , alpha in zip(delta_weights, alphas):\n",
    "            for param in delta.values():\n",
    "                param.data = alpha * param.data  # refers to delta_w_i * alpha_i\n",
    "    \n",
    "        # aggregate model params \n",
    "        alpha_delta = OrderedDict() \n",
    "        for delta in delta_weights: ## it's aggregating delta_w_i * alpha_i\n",
    "            for layer_name, param in delta.items():\n",
    "                if layer_name not in alpha_delta:  \n",
    "                    alpha_delta[layer_name] = param\n",
    "                else:\n",
    "                    alpha_delta[layer_name] += param\n",
    "    return alpha_delta\n",
    "    # with torch.no_grad():\n",
    "    # # calculate weights\n",
    "    #     weight_list = [model.parameters() for model in models]\n",
    "    #     weight_sum = sum(weight_list)\n",
    "    #     # weight normalization\n",
    "    #     weight_list = list(map(lambda w: w / weight_sum, weight_list))  \n",
    "    #     avg_model = SimpleCNN()\n",
    "    #     num_clients = len(models)\n",
    "        \n",
    "    #     for param_avg , *params in zip(avg_model.parameters(), *[model.parameters() for model in models]):\n",
    "    #         param_avg.data = torch.mean(torch.stack(params), dim=0)\n",
    "    # return avg_model   \n",
    "\n",
    "\n",
    "\n",
    "########################################### global federated training ##########################################################\n",
    "        \n",
    "\n",
    "# Function to simulate federated learning with non-IID and heterogeneous data\n",
    "def federated_train(dataset, algo, model_name, num_clients, global_epochs, batch_size, lr , global_lr, train_set, test_set, \n",
    "                               local_epochs , split_layer, mix_factor , alpha, beta, criterion, frac):\n",
    "    scale = \"large_scale\" if num_clients > 20 else \"small_scale\"\n",
    "    MODEL = get_model(model_name)\n",
    "    num_classes = get_num_classes(dataset)\n",
    "    #create global model\n",
    "    global_model = MODEL[dataset](num_classes)\n",
    "    \n",
    "# there shouldn't be a global optimizer ! we don't have any data to train on them\n",
    "# global_optimizer = optim.Adam(global_model.parameters() , lr= global_lr)\n",
    "    \n",
    "# Create models and optimizers for each client\n",
    "    models = [MODEL[dataset](num_classes) for _ in range(num_clients)]\n",
    "# multiparallelism     \n",
    "    # if torch.cuda.device_count() > 1:\n",
    "    #     models =  [nn.DataParallel(model) for model in models]\n",
    "    #     global_model = nn.DataParallel(global_model)\n",
    "    #     print(f\"Using {torch.cuda.device_count()} GPUs\")\n",
    "        \n",
    "        \n",
    "    # optimizers =  [None] * num_clients\n",
    "        \n",
    "    if algo == \"pMixFed\":\n",
    "        for i in range(num_clients):\n",
    "            print(f\"!!local training started for client {i}\")\n",
    "        optimizers = [optim.Adam(model.parameters(), lr=lr) for model in models]\n",
    "            \n",
    "    elif algo == \"FedBABU\":\n",
    "        global_model.BABU_split()\n",
    "        for i in range(num_clients):\n",
    "            print(f\"local training started for client {i}\")\n",
    "            models[i].BABU_split()\n",
    "            # global_optimizer = optim.SGD([{'params': models[i].body_params, 'lr': lr},\n",
    "            #                                  {'params': models[i].head_params, 'lr': 0. }])\n",
    "        optimizers = [optim.SGD([{'params': models[i].global_params.values(), 'lr': lr , 'name': \"body\" },\n",
    "                                {'params': models[i].local_params.values(), 'lr': 0. , \"name\": \"head\" }]) for model in models]\n",
    "\n",
    "\n",
    "    elif algo == \"FedAlt\" or \"FedSim\" or \"pFedMix\":\n",
    "        global_model.split(split_layer)\n",
    "        for i in range(num_clients):\n",
    "            print(f\"local training started for client {i}\")\n",
    "            models[i].split(split_layer) ## ??? we can later change the spli_layer \n",
    "    # for every client ( model heterogenity) and also learn it adaptively through training)\n",
    "        optimizers = [optim.Adam(model.parameters(), lr=lr) for model in models]\n",
    "    \n",
    "\n",
    "    # this is the list of the [avg of the local clients lossess]\n",
    "    Final_train_loss = []\n",
    "    Final_train_accuracy = []\n",
    "    Final_test_accuracy = []\n",
    "    for epoch in range(global_epochs):\n",
    "        # Train models on client datasets\n",
    "        alphas = []\n",
    "        delta_weights = []\n",
    "        global_loss = []\n",
    "        train_accuracy = []\n",
    "        test_accuracy = []\n",
    "        random_client_selection = random.choices(range(num_clients), k=frac)\n",
    "        for i in random_client_selection:\n",
    "            train_loader = DataLoader(train_set[i], batch_size=batch_size, shuffle=True)\n",
    "            test_loader = DataLoader(test_set[i], batch_size=batch_size, shuffle=True)\n",
    "            alpha = len(train_loader.dataset)\n",
    "            client_loss, client_accuracy_train, client_accuracy_test, delta_weight = train_model(algo, global_model ,models[i], train_loader, test_loader, optimizers[i], criterion , local_epochs , lr , \n",
    "                                                    mix_factor)\n",
    "            delta_weights.append(delta_weight)\n",
    "            global_loss.append(client_loss)\n",
    "            train_accuracy.append(client_accuracy_train)\n",
    "            test_accuracy.append(client_accuracy_test)\n",
    "            alphas.append(alpha)\n",
    "\n",
    "        \n",
    "        avg_global_loss = sum (global_loss) / len(global_loss) \n",
    "        Final_train_loss.append(avg_global_loss.detach().cpu().numpy())\n",
    "        avg_train_accuracy = sum (train_accuracy) / len(train_accuracy)\n",
    "        Final_train_accuracy.append(avg_train_accuracy.detach().cpu().numpy())\n",
    "        avg_test_accuracy = sum (test_accuracy) / len(test_accuracy)\n",
    "        Final_test_accuracy.append(avg_test_accuracy.detach().cpu().numpy())\n",
    "        \n",
    "        print(f'***** round {epoch} /// loss : {avg_global_loss} , train_acc : {avg_train_accuracy} , test_acc : {avg_test_accuracy} *******')\n",
    "\n",
    " \n",
    "        torch.cuda.empty_cache()\n",
    "                                                          \n",
    "        # Perform federated averaging\n",
    "        avg_delta_weight = federated_averaging(delta_weights , alphas , global_lr)\n",
    "\n",
    "\n",
    "        # update global model\n",
    "        global_model_params = OrderedDict(global_model.named_parameters())\n",
    "        \n",
    "        for layer_name, delta in global_model_params.items():\n",
    "            global_model_params[layer_name] = global_model_params[layer_name].to(device)\n",
    "\n",
    "        if algo == \"pFedMix\" or \"pMixFed\":\n",
    "            with torch.no_grad():\n",
    "                for layer_name, delta in avg_delta_weight.items():\n",
    "                    global_model_params[layer_name]= delta\n",
    "            \n",
    "        else: ##  elif (algo == \"FedAvg\" or 'FedAlt\": \n",
    "            with torch.no_grad():\n",
    "                for layer_name, delta in avg_delta_weight.items():\n",
    "                    global_model_params[layer_name].sub_(global_lr * delta)  # w(t+1) = w(t) - (LR * delta_w(t))\n",
    "                    \n",
    "        global_model.load_state_dict(global_model_params, strict =False)         \n",
    "\n",
    "        print(f\"Epoch {epoch + 1}/{global_epochs} completed\")\n",
    "\n",
    "    os.makedirs(f'Results', exist_ok=True)\n",
    "    with open(f'Results/results{algo}-{lr}-{scale}-{model_name}-train.pkl', 'wb') as file:\n",
    "        pickle.dump(\n",
    "            {\n",
    "                    \"Final_train_accuracy\": Final_train_accuracy,\n",
    "                    \"Final_test_accuracy\": Final_test_accuracy,\n",
    "                    \"Final_train_loss\" : Final_train_loss,\n",
    "                    \"global_model\": global_model,\n",
    "                    \"local_models\": models,\n",
    "                    \"optimizers\" : optimizers,\n",
    "                }, file\n",
    "        )\n",
    "\n",
    "\n",
    "    # Plot the training loss\n",
    "    plt.plot(Final_train_accuracy, label='overall Training accuracy')\n",
    "    plt.xlabel('communication round')\n",
    "    plt.ylabel('Accuracy')\n",
    "    plt.title('overall Personalziation accuracy in each round of global training')\n",
    "    plt.legend()\n",
    "    plt.show()\n",
    "\n",
    "    return Final_train_loss , Final_train_accuracy , Final_test_accuracy\n",
    "        \n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d2a23542-9ad4-4d3a-9eb3-0602f3436b54",
   "metadata": {},
   "source": [
    "#"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0ab07a4e-5329-463c-837f-98db8ac9839c",
   "metadata": {},
   "source": [
    "## Evaluation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "7e32ff3a-95dd-409b-8f21-c267ac059590",
   "metadata": {},
   "outputs": [],
   "source": [
    "def federated_test(algo, model_name, num_clients, dataset, test_sets, criterion, local_epochs, beta, lr):\n",
    "    \n",
    "    scale = \"large_scale\" if num_clients > 20 else \"small_scale\"\n",
    "    with open(f'Results/results{algo}-{lr}-{scale}-{model_name}-train.pkl', \"rb\") as file:\n",
    "            data = pickle.load(file)\n",
    "    global_model = data[\"global_model\"]\n",
    "    models= data[\"local_models\"]\n",
    "    optimizers = data[\"optimizers\"]\n",
    "    \n",
    "\n",
    "    # global_params_dict = OrderedDict()\n",
    "    # for name, param in model.global_params.items():\n",
    "    #     global_params_dict[name] = param.detach().clone().data\n",
    "\n",
    "    clients_accuracy_before = []\n",
    "    clients_total_loss_before = []\n",
    "    clients_accuracy_after = []\n",
    "    clients_total_loss_after = []\n",
    "\n",
    "    alphas = []\n",
    "    for i in range(num_clients):\n",
    "        test_loader = DataLoader(test_sets[i], batch_size=batch_size, shuffle=True)\n",
    "        # accuracy_after , total_loss_after , accuracy_before , total_loss_befor = 0\n",
    "        accuracy_after , total_loss_after , accuracy_before , total_loss_before = test_local_model(algo, \n",
    "                                                                                                   global_model, \n",
    "                                                                                                   models[i] , \n",
    "                                                                                                   test_loader, \n",
    "                                                                                                   optimizers[i], \n",
    "                                                                                                   criterion, local_epochs)\n",
    "        alphas.append(len(test_loader.dataset)) #???? what is the use of alpha here?\n",
    "        clients_accuracy_before.append(accuracy_before)\n",
    "        clients_total_loss_before.append(total_loss_before) \n",
    "        clients_accuracy_after.append(accuracy_after)\n",
    "        clients_total_loss_after.append(total_loss_after)\n",
    "        print(f\"************** evaluation result for client {i} : loss {clients_total_loss_before[i]}-->{clients_total_loss_after[i]}********************\")\n",
    "        print(f'************** accuracy {clients_accuracy_before[i]}%-->{clients_accuracy_after[i]}% ****************************************')\n",
    "        \n",
    "    result = [clients_accuracy_before, clients_total_loss_before, clients_accuracy_after, clients_total_loss_after]\n",
    "    final_test_result = [sum(out)/num_clients for out in result]\n",
    "\n",
    "\n",
    "    with open(f'Results/Results/results{algo}-{lr}-{scale}-{model_name}-test.pkl', 'wb') as file:\n",
    "        pickle.dump(\n",
    "            {\n",
    "                    \"clients_accuracy_before\": final_test_result[0],\n",
    "                    \"clients_total_loss_before\" : final_test_result[1],\n",
    "                    \"clients_accuracy_after\": final_test_result[2],\n",
    "                    \"clients_total_loss_after\": final_test_result[3],\n",
    "                }, file\n",
    "        )\n",
    "     \n",
    "    return final_test_result\n",
    "\n",
    "def test_local_model(algo, global_model, model , test_loader, optimizer, criterion, local_epochs):\n",
    "    global_model.to(device)\n",
    "    model.to(device)\n",
    "    global_params_dict = OrderedDict()\n",
    "    for name, param in model.global_params.items():\n",
    "        global_params_dict[name] = param.detach().clone().data\n",
    "        \n",
    "    if algo == \"FedAvg\":\n",
    "        global_model_params = deepcopy(OrderedDict(global_model.named_parameters()))\n",
    "        model.load_state_dict(global_model_params, strict=True)\n",
    "        # Evaluation before the local training\n",
    "    elif algo == \"pMixFed\":\n",
    "        Lm_mixup = layerwise_mixup(global_model, model,  mix_factor , phase='broadcast')\n",
    "        model.load_state_dict(Lm_mixup, strict=False)\n",
    "        delta_weight =  global_params_dict\n",
    "    \n",
    "    elif algo == \"pFedMix\":\n",
    "        Lm_mixup = Mixup_global_to_local(global_model.global_params, model.global_params,  mix_factor )\n",
    "        model.load_state_dict(Lm_mixup, strict=False)\n",
    "        model.load_state_dict (model.local_params , strict=False)\n",
    "        delta_weight =  global_params_dict \n",
    "        # Evaluation before the local training\n",
    "        \n",
    "    elif algo == \"FedAlt\" or \"FedSim\":\n",
    "        ## it could be changed for cold-strat-users to global model \n",
    "        local_model_params = deepcopy(OrderedDict(model.named_parameters())) \n",
    "        model.load_state_dict(local_model_params, strict=False)\n",
    "        delta_weight =  global_params_dict\n",
    "        # Evaluation before the local training\n",
    "        \n",
    "    else:\n",
    "        raise ValueError(f'the algorithm {algo} is not supported!')\n",
    "    accuracy_before , total_loss_before = evaluate(test_loader , model , criterion)    \n",
    "    # train on test loader for local inner epochs\n",
    "    train(test_loader, model , criterion, algo , optimizer )\n",
    "\n",
    "    \n",
    "    # Evaluation after the local training\n",
    "    accuracy_after , total_loss_after = evaluate(test_loader , model , criterion) \n",
    "    return accuracy_after , total_loss_after , accuracy_before , total_loss_before\n",
    "    \n",
    "\n",
    "    \n",
    "        \n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "753a66ba-34a4-48f7-a2d7-0439d8f4ba4a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "global params are layer1.0.conv1.weight\n",
      "global params are layer1.0.conv1.bias\n",
      "global params are layer1.0.bn1.weight\n",
      "global params are layer1.0.bn1.bias\n",
      "global params are layer2.0.conv1.weight\n",
      "global params are layer2.0.conv1.bias\n",
      "global params are layer2.0.bn1.weight\n",
      "global params are layer2.0.bn1.bias\n",
      "local training started for client 0\n",
      "global params are layer1.0.conv1.weight\n",
      "global params are layer1.0.conv1.bias\n",
      "global params are layer1.0.bn1.weight\n",
      "global params are layer1.0.bn1.bias\n",
      "global params are layer2.0.conv1.weight\n",
      "global params are layer2.0.conv1.bias\n",
      "global params are layer2.0.bn1.weight\n",
      "global params are layer2.0.bn1.bias\n",
      "local training started for client 1\n",
      "global params are layer1.0.conv1.weight\n",
      "global params are layer1.0.conv1.bias\n",
      "global params are layer1.0.bn1.weight\n",
      "global params are layer1.0.bn1.bias\n",
      "global params are layer2.0.conv1.weight\n",
      "global params are layer2.0.conv1.bias\n",
      "global params are layer2.0.bn1.weight\n",
      "global params are layer2.0.bn1.bias\n",
      "local training started for client 2\n",
      "global params are layer1.0.conv1.weight\n",
      "global params are layer1.0.conv1.bias\n",
      "global params are layer1.0.bn1.weight\n",
      "global params are layer1.0.bn1.bias\n",
      "global params are layer2.0.conv1.weight\n",
      "global params are layer2.0.conv1.bias\n",
      "global params are layer2.0.bn1.weight\n",
      "global params are layer2.0.bn1.bias\n",
      "local training started for client 3\n",
      "global params are layer1.0.conv1.weight\n",
      "global params are layer1.0.conv1.bias\n",
      "global params are layer1.0.bn1.weight\n",
      "global params are layer1.0.bn1.bias\n",
      "global params are layer2.0.conv1.weight\n",
      "global params are layer2.0.conv1.bias\n",
      "global params are layer2.0.bn1.weight\n",
      "global params are layer2.0.bn1.bias\n",
      "local training started for client 4\n",
      "global params are layer1.0.conv1.weight\n",
      "global params are layer1.0.conv1.bias\n",
      "global params are layer1.0.bn1.weight\n",
      "global params are layer1.0.bn1.bias\n",
      "global params are layer2.0.conv1.weight\n",
      "global params are layer2.0.conv1.bias\n",
      "global params are layer2.0.bn1.weight\n",
      "global params are layer2.0.bn1.bias\n",
      "local training started for client 5\n",
      "global params are layer1.0.conv1.weight\n",
      "global params are layer1.0.conv1.bias\n",
      "global params are layer1.0.bn1.weight\n",
      "global params are layer1.0.bn1.bias\n",
      "global params are layer2.0.conv1.weight\n",
      "global params are layer2.0.conv1.bias\n",
      "global params are layer2.0.bn1.weight\n",
      "global params are layer2.0.bn1.bias\n",
      "local training started for client 6\n",
      "global params are layer1.0.conv1.weight\n",
      "global params are layer1.0.conv1.bias\n",
      "global params are layer1.0.bn1.weight\n",
      "global params are layer1.0.bn1.bias\n",
      "global params are layer2.0.conv1.weight\n",
      "global params are layer2.0.conv1.bias\n",
      "global params are layer2.0.bn1.weight\n",
      "global params are layer2.0.bn1.bias\n",
      "local training started for client 7\n",
      "global params are layer1.0.conv1.weight\n",
      "global params are layer1.0.conv1.bias\n",
      "global params are layer1.0.bn1.weight\n",
      "global params are layer1.0.bn1.bias\n",
      "global params are layer2.0.conv1.weight\n",
      "global params are layer2.0.conv1.bias\n",
      "global params are layer2.0.bn1.weight\n",
      "global params are layer2.0.bn1.bias\n",
      "local training started for client 8\n",
      "global params are layer1.0.conv1.weight\n",
      "global params are layer1.0.conv1.bias\n",
      "global params are layer1.0.bn1.weight\n",
      "global params are layer1.0.bn1.bias\n",
      "global params are layer2.0.conv1.weight\n",
      "global params are layer2.0.conv1.bias\n",
      "global params are layer2.0.bn1.weight\n",
      "global params are layer2.0.bn1.bias\n",
      "local training started for client 9\n",
      "global params are layer1.0.conv1.weight\n",
      "global params are layer1.0.conv1.bias\n",
      "global params are layer1.0.bn1.weight\n",
      "global params are layer1.0.bn1.bias\n",
      "global params are layer2.0.conv1.weight\n",
      "global params are layer2.0.conv1.bias\n",
      "global params are layer2.0.bn1.weight\n",
      "global params are layer2.0.bn1.bias\n",
      "***** round 0 /// loss : 0.032399993389844894 , train_acc : 65.21327209472656 , test_acc : 24.034595489501953 *******\n",
      "Epoch 1/100 completed\n",
      "***** round 1 /// loss : 0.044702012091875076 , train_acc : 50.521297454833984 , test_acc : 13.121310234069824 *******\n",
      "Epoch 2/100 completed\n",
      "***** round 2 /// loss : 0.03211633116006851 , train_acc : 63.89522171020508 , test_acc : 19.7288818359375 *******\n",
      "Epoch 3/100 completed\n",
      "***** round 3 /// loss : 0.03658142313361168 , train_acc : 59.87268829345703 , test_acc : 18.55536460876465 *******\n",
      "Epoch 4/100 completed\n",
      "***** round 4 /// loss : 0.026857275515794754 , train_acc : 71.00401306152344 , test_acc : 15.970707893371582 *******\n",
      "Epoch 5/100 completed\n",
      "***** round 5 /// loss : 0.027183106169104576 , train_acc : 69.78646850585938 , test_acc : 10.249676704406738 *******\n",
      "Epoch 6/100 completed\n",
      "***** round 6 /// loss : 0.026393307372927666 , train_acc : 71.436279296875 , test_acc : 20.143957138061523 *******\n",
      "Epoch 7/100 completed\n",
      "***** round 7 /// loss : 0.026727020740509033 , train_acc : 69.4384994506836 , test_acc : 17.207746505737305 *******\n",
      "Epoch 8/100 completed\n",
      "***** round 8 /// loss : 0.024891570210456848 , train_acc : 73.0984878540039 , test_acc : 24.25477409362793 *******\n",
      "Epoch 9/100 completed\n",
      "***** round 9 /// loss : 0.025679394602775574 , train_acc : 71.50056457519531 , test_acc : 23.5012264251709 *******\n",
      "Epoch 10/100 completed\n",
      "***** round 10 /// loss : 0.022680608555674553 , train_acc : 74.36954498291016 , test_acc : 22.774133682250977 *******\n",
      "Epoch 11/100 completed\n",
      "***** round 11 /// loss : 0.022587379440665245 , train_acc : 74.89156341552734 , test_acc : 20.189603805541992 *******\n",
      "Epoch 12/100 completed\n",
      "***** round 12 /// loss : 0.023401862010359764 , train_acc : 74.56106567382812 , test_acc : 33.43561935424805 *******\n",
      "Epoch 13/100 completed\n",
      "***** round 13 /// loss : 0.021494926884770393 , train_acc : 76.34017944335938 , test_acc : 30.578088760375977 *******\n",
      "Epoch 14/100 completed\n",
      "***** round 14 /// loss : 0.0210284236818552 , train_acc : 76.77230072021484 , test_acc : 26.210342407226562 *******\n",
      "Epoch 15/100 completed\n",
      "***** round 15 /// loss : 0.019406979903578758 , train_acc : 78.56157684326172 , test_acc : 20.23712730407715 *******\n",
      "Epoch 16/100 completed\n",
      "***** round 16 /// loss : 0.01834208332002163 , train_acc : 80.08012390136719 , test_acc : 22.431596755981445 *******\n",
      "Epoch 17/100 completed\n",
      "***** round 17 /// loss : 0.020896922796964645 , train_acc : 76.58216094970703 , test_acc : 31.13958740234375 *******\n",
      "Epoch 18/100 completed\n",
      "***** round 18 /// loss : 0.019387969747185707 , train_acc : 78.93413543701172 , test_acc : 26.463361740112305 *******\n",
      "Epoch 19/100 completed\n",
      "***** round 19 /// loss : 0.018751850351691246 , train_acc : 79.3282699584961 , test_acc : 32.596473693847656 *******\n",
      "Epoch 20/100 completed\n",
      "***** round 20 /// loss : 0.018368443474173546 , train_acc : 79.98387908935547 , test_acc : 25.69257926940918 *******\n",
      "Epoch 21/100 completed\n",
      "***** round 21 /// loss : 0.01651291735470295 , train_acc : 82.112060546875 , test_acc : 26.408132553100586 *******\n",
      "Epoch 22/100 completed\n",
      "***** round 22 /// loss : 0.018997929990291595 , train_acc : 79.05254364013672 , test_acc : 32.298606872558594 *******\n",
      "Epoch 23/100 completed\n",
      "***** round 23 /// loss : 0.016314629465341568 , train_acc : 82.38338470458984 , test_acc : 23.501123428344727 *******\n",
      "Epoch 24/100 completed\n",
      "***** round 24 /// loss : 0.01476247888058424 , train_acc : 84.49050903320312 , test_acc : 25.708410263061523 *******\n",
      "Epoch 25/100 completed\n",
      "***** round 25 /// loss : 0.014799010939896107 , train_acc : 84.46074676513672 , test_acc : 27.955678939819336 *******\n",
      "Epoch 26/100 completed\n",
      "***** round 26 /// loss : 0.018402155488729477 , train_acc : 80.30692291259766 , test_acc : 30.577871322631836 *******\n",
      "Epoch 27/100 completed\n",
      "***** round 27 /// loss : 0.014658818021416664 , train_acc : 84.30982208251953 , test_acc : 18.373708724975586 *******\n",
      "Epoch 28/100 completed\n",
      "***** round 28 /// loss : 0.012378483079373837 , train_acc : 87.43547058105469 , test_acc : 28.71286964416504 *******\n",
      "Epoch 29/100 completed\n",
      "***** round 29 /// loss : 0.012549842707812786 , train_acc : 86.7955551147461 , test_acc : 38.16665267944336 *******\n",
      "Epoch 30/100 completed\n",
      "***** round 30 /// loss : 0.017154602333903313 , train_acc : 81.09273529052734 , test_acc : 31.227869033813477 *******\n",
      "Epoch 31/100 completed\n",
      "***** round 31 /// loss : 0.013480834662914276 , train_acc : 86.202392578125 , test_acc : 25.52398109436035 *******\n",
      "Epoch 32/100 completed\n",
      "***** round 32 /// loss : 0.01726534403860569 , train_acc : 82.135498046875 , test_acc : 38.822322845458984 *******\n",
      "Epoch 33/100 completed\n",
      "***** round 33 /// loss : 0.014589783735573292 , train_acc : 85.29145812988281 , test_acc : 31.637603759765625 *******\n",
      "Epoch 34/100 completed\n",
      "***** round 34 /// loss : 0.01615010015666485 , train_acc : 84.19380950927734 , test_acc : 22.90577507019043 *******\n",
      "Epoch 35/100 completed\n",
      "***** round 35 /// loss : 0.01460632961243391 , train_acc : 84.37214660644531 , test_acc : 19.249767303466797 *******\n",
      "Epoch 36/100 completed\n",
      "***** round 36 /// loss : 0.01655236817896366 , train_acc : 83.3502426147461 , test_acc : 26.362411499023438 *******\n",
      "Epoch 37/100 completed\n",
      "***** round 37 /// loss : 0.017897052690386772 , train_acc : 81.87403106689453 , test_acc : 26.48258399963379 *******\n",
      "Epoch 38/100 completed\n",
      "***** round 38 /// loss : 0.01422621589154005 , train_acc : 85.5288314819336 , test_acc : 23.016077041625977 *******\n",
      "Epoch 39/100 completed\n",
      "***** round 39 /// loss : 0.01326336432248354 , train_acc : 86.44918060302734 , test_acc : 27.873523712158203 *******\n",
      "Epoch 40/100 completed\n",
      "***** round 40 /// loss : 0.012887316755950451 , train_acc : 86.81768798828125 , test_acc : 27.134170532226562 *******\n",
      "Epoch 41/100 completed\n",
      "***** round 41 /// loss : 0.010505358688533306 , train_acc : 89.38542938232422 , test_acc : 32.16542434692383 *******\n",
      "Epoch 42/100 completed\n",
      "***** round 42 /// loss : 0.008421097882091999 , train_acc : 91.4100112915039 , test_acc : 26.688613891601562 *******\n",
      "Epoch 43/100 completed\n",
      "***** round 43 /// loss : 0.011193699203431606 , train_acc : 88.20741271972656 , test_acc : 30.61557960510254 *******\n",
      "Epoch 44/100 completed\n",
      "***** round 44 /// loss : 0.011609815061092377 , train_acc : 87.64817810058594 , test_acc : 34.544151306152344 *******\n",
      "Epoch 45/100 completed\n",
      "***** round 45 /// loss : 0.009439225308597088 , train_acc : 91.25086212158203 , test_acc : 28.323083877563477 *******\n",
      "Epoch 46/100 completed\n",
      "***** round 46 /// loss : 0.0085283312946558 , train_acc : 91.7315444946289 , test_acc : 27.185876846313477 *******\n",
      "Epoch 47/100 completed\n",
      "***** round 47 /// loss : 0.00897594727575779 , train_acc : 91.0262680053711 , test_acc : 32.56919860839844 *******\n",
      "Epoch 48/100 completed\n",
      "***** round 48 /// loss : 0.010103601031005383 , train_acc : 89.8751449584961 , test_acc : 32.698490142822266 *******\n",
      "Epoch 49/100 completed\n",
      "***** round 49 /// loss : 0.005972431972622871 , train_acc : 94.07125091552734 , test_acc : 22.321815490722656 *******\n",
      "Epoch 50/100 completed\n",
      "***** round 50 /// loss : 0.006854643579572439 , train_acc : 93.11998748779297 , test_acc : 27.657888412475586 *******\n",
      "Epoch 51/100 completed\n",
      "***** round 51 /// loss : 0.007067470345646143 , train_acc : 92.73552703857422 , test_acc : 34.558040618896484 *******\n",
      "Epoch 52/100 completed\n",
      "***** round 52 /// loss : 0.007246391382068396 , train_acc : 92.72057342529297 , test_acc : 23.405471801757812 *******\n",
      "Epoch 53/100 completed\n",
      "***** round 53 /// loss : 0.008772182278335094 , train_acc : 91.47823333740234 , test_acc : 19.83147621154785 *******\n",
      "Epoch 54/100 completed\n",
      "***** round 54 /// loss : 0.01140937115997076 , train_acc : 89.77631378173828 , test_acc : 27.727426528930664 *******\n",
      "Epoch 55/100 completed\n",
      "***** round 55 /// loss : 0.013772556558251381 , train_acc : 86.78531646728516 , test_acc : 21.40789794921875 *******\n",
      "Epoch 56/100 completed\n",
      "***** round 56 /// loss : 0.013565647415816784 , train_acc : 86.887451171875 , test_acc : 29.10605239868164 *******\n",
      "Epoch 57/100 completed\n",
      "***** round 57 /// loss : 0.010783023200929165 , train_acc : 90.58834075927734 , test_acc : 35.1627311706543 *******\n",
      "Epoch 58/100 completed\n",
      "***** round 58 /// loss : 0.010336589999496937 , train_acc : 91.13240051269531 , test_acc : 33.3633918762207 *******\n",
      "Epoch 59/100 completed\n",
      "***** round 59 /// loss : 0.01590322144329548 , train_acc : 84.86753845214844 , test_acc : 29.655012130737305 *******\n",
      "Epoch 60/100 completed\n",
      "***** round 60 /// loss : 0.009488359093666077 , train_acc : 91.89678192138672 , test_acc : 32.625545501708984 *******\n",
      "Epoch 61/100 completed\n",
      "***** round 61 /// loss : 0.021840257570147514 , train_acc : 76.9635238647461 , test_acc : 22.54080581665039 *******\n",
      "Epoch 62/100 completed\n",
      "***** round 62 /// loss : 0.01690669357776642 , train_acc : 82.4383316040039 , test_acc : 17.29311180114746 *******\n",
      "Epoch 63/100 completed\n",
      "***** round 63 /// loss : 0.016223622485995293 , train_acc : 82.61958312988281 , test_acc : 17.00143814086914 *******\n",
      "Epoch 64/100 completed\n",
      "***** round 64 /// loss : 0.019534748047590256 , train_acc : 80.07551574707031 , test_acc : 28.854206085205078 *******\n",
      "Epoch 65/100 completed\n",
      "***** round 65 /// loss : 0.014161531813442707 , train_acc : 85.52465057373047 , test_acc : 28.11476707458496 *******\n",
      "Epoch 66/100 completed\n",
      "***** round 66 /// loss : 0.015643475577235222 , train_acc : 83.74740600585938 , test_acc : 25.5708065032959 *******\n",
      "Epoch 67/100 completed\n",
      "***** round 67 /// loss : 0.011206094175577164 , train_acc : 88.88714599609375 , test_acc : 26.698144912719727 *******\n",
      "Epoch 68/100 completed\n",
      "***** round 68 /// loss : 0.009220103733241558 , train_acc : 90.7442626953125 , test_acc : 25.41427993774414 *******\n",
      "Epoch 69/100 completed\n",
      "***** round 69 /// loss : 0.008648205548524857 , train_acc : 91.56489562988281 , test_acc : 29.279388427734375 *******\n",
      "Epoch 70/100 completed\n",
      "***** round 70 /// loss : 0.013192686252295971 , train_acc : 86.41143798828125 , test_acc : 23.60357093811035 *******\n",
      "Epoch 71/100 completed\n",
      "***** round 71 /// loss : 0.007412967272102833 , train_acc : 93.46675872802734 , test_acc : 29.843231201171875 *******\n",
      "Epoch 72/100 completed\n",
      "***** round 72 /// loss : 0.009773476980626583 , train_acc : 90.20944213867188 , test_acc : 33.28553009033203 *******\n",
      "Epoch 73/100 completed\n",
      "***** round 73 /// loss : 0.008461741730570793 , train_acc : 91.85533905029297 , test_acc : 24.879005432128906 *******\n",
      "Epoch 74/100 completed\n",
      "***** round 74 /// loss : 0.009884083643555641 , train_acc : 90.50852966308594 , test_acc : 28.814237594604492 *******\n",
      "Epoch 75/100 completed\n",
      "***** round 75 /// loss : 0.008818811737000942 , train_acc : 91.32831573486328 , test_acc : 25.31781768798828 *******\n",
      "Epoch 76/100 completed\n",
      "***** round 76 /// loss : 0.006483821664005518 , train_acc : 93.73080444335938 , test_acc : 27.720335006713867 *******\n",
      "Epoch 77/100 completed\n",
      "***** round 77 /// loss : 0.0070990994572639465 , train_acc : 92.8464584350586 , test_acc : 26.645227432250977 *******\n",
      "Epoch 78/100 completed\n",
      "***** round 78 /// loss : 0.008086106739938259 , train_acc : 91.90900421142578 , test_acc : 34.56362533569336 *******\n",
      "Epoch 79/100 completed\n",
      "***** round 79 /// loss : 0.008220464922487736 , train_acc : 91.69178771972656 , test_acc : 38.282161712646484 *******\n",
      "Epoch 80/100 completed\n",
      "***** round 80 /// loss : 0.00609819358214736 , train_acc : 93.888916015625 , test_acc : 31.086095809936523 *******\n",
      "Epoch 81/100 completed\n",
      "***** round 81 /// loss : 0.004647416528314352 , train_acc : 95.30789947509766 , test_acc : 31.186492919921875 *******\n",
      "Epoch 82/100 completed\n",
      "***** round 82 /// loss : 0.006497369613498449 , train_acc : 93.45507049560547 , test_acc : 37.85761642456055 *******\n",
      "Epoch 83/100 completed\n",
      "***** round 83 /// loss : 0.003797798650339246 , train_acc : 96.26873016357422 , test_acc : 30.9271240234375 *******\n",
      "Epoch 84/100 completed\n",
      "***** round 84 /// loss : 0.005793004762381315 , train_acc : 94.39950561523438 , test_acc : 30.6899356842041 *******\n",
      "Epoch 85/100 completed\n",
      "***** round 85 /// loss : 0.00331994891166687 , train_acc : 97.4946060180664 , test_acc : 36.69088363647461 *******\n",
      "Epoch 86/100 completed\n",
      "***** round 86 /// loss : 0.006222335621714592 , train_acc : 94.79533386230469 , test_acc : 29.464115142822266 *******\n",
      "Epoch 87/100 completed\n",
      "***** round 87 /// loss : 0.0038559853564947844 , train_acc : 96.04410552978516 , test_acc : 28.834075927734375 *******\n",
      "Epoch 88/100 completed\n",
      "***** round 88 /// loss : 0.002796510001644492 , train_acc : 97.18208312988281 , test_acc : 28.926956176757812 *******\n",
      "Epoch 89/100 completed\n",
      "***** round 89 /// loss : 0.0031347272451967 , train_acc : 96.88652801513672 , test_acc : 28.074140548706055 *******\n",
      "Epoch 90/100 completed\n",
      "***** round 90 /// loss : 0.00137619161978364 , train_acc : 98.60799407958984 , test_acc : 27.81488609313965 *******\n",
      "Epoch 91/100 completed\n",
      "***** round 91 /// loss : 0.0011567765614017844 , train_acc : 98.91478729248047 , test_acc : 24.531049728393555 *******\n",
      "Epoch 92/100 completed\n",
      "***** round 92 /// loss : 0.0019183530239388347 , train_acc : 98.28020477294922 , test_acc : 30.38385009765625 *******\n",
      "Epoch 93/100 completed\n",
      "***** round 93 /// loss : 0.003011491149663925 , train_acc : 97.08609008789062 , test_acc : 28.844308853149414 *******\n",
      "Epoch 94/100 completed\n",
      "***** round 94 /// loss : 0.003823010716587305 , train_acc : 96.60222625732422 , test_acc : 29.677061080932617 *******\n",
      "Epoch 95/100 completed\n",
      "***** round 95 /// loss : 0.002166813937947154 , train_acc : 98.54572296142578 , test_acc : 34.92060470581055 *******\n",
      "Epoch 96/100 completed\n",
      "***** round 96 /// loss : 0.0020529155153781176 , train_acc : 98.19477081298828 , test_acc : 32.967166900634766 *******\n",
      "Epoch 97/100 completed\n",
      "***** round 97 /// loss : 0.0016087808180600405 , train_acc : 98.64716339111328 , test_acc : 28.032236099243164 *******\n",
      "Epoch 98/100 completed\n",
      "***** round 98 /// loss : 0.0017238733125850558 , train_acc : 98.442138671875 , test_acc : 29.09195899963379 *******\n",
      "Epoch 99/100 completed\n",
      "***** round 99 /// loss : 0.0005858070799149573 , train_acc : 99.55863189697266 , test_acc : 34.41685104370117 *******\n",
      "Epoch 100/100 completed\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "ename": "TypeError",
     "evalue": "federated_test() missing 1 required positional argument: 'lr'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mTypeError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[8], line 49\u001b[0m\n\u001b[1;32m     41\u001b[0m Final_train_loss , Final_train_accuracy , Final_test_accuracy \u001b[38;5;241m=\u001b[39m federated_train(dataset, algo, model , num_clients, global_epochs, batch_size, lr , global_lr, \n\u001b[1;32m     42\u001b[0m                             train_set, test_set, local_epochs, split_layer , mix_factor , alpha, beta, criterion , frac)\n\u001b[1;32m     47\u001b[0m  \u001b[38;5;66;03m############################################## test ###################################################### \u001b[39;00m\n\u001b[0;32m---> 49\u001b[0m final_test_result \u001b[38;5;241m=\u001b[39m \u001b[43mfederated_test\u001b[49m\u001b[43m(\u001b[49m\u001b[43malgo\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mmodel\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mnum_clients\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mdataset\u001b[49m\u001b[43m \u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtest_set\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mcriterion\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mlocal_epochs\u001b[49m\u001b[43m \u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mbeta\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     50\u001b[0m \u001b[38;5;28mprint\u001b[39m (\u001b[38;5;124mf\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124m clients_accuracy_before : \u001b[39m\u001b[38;5;132;01m{\u001b[39;00mfinal_test_result[\u001b[38;5;241m0\u001b[39m]\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m'\u001b[39m\n\u001b[1;32m     51\u001b[0m         \u001b[38;5;124mf\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;130;01m\\n\u001b[39;00m\u001b[38;5;124m clients_total_loss_before: \u001b[39m\u001b[38;5;132;01m{\u001b[39;00mfinal_test_result[\u001b[38;5;241m1\u001b[39m]\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m'\u001b[39m\n\u001b[1;32m     52\u001b[0m         \u001b[38;5;124mf\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;130;01m\\n\u001b[39;00m\u001b[38;5;124m clients_accuracy_after : \u001b[39m\u001b[38;5;132;01m{\u001b[39;00mfinal_test_result[\u001b[38;5;241m2\u001b[39m]\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m'\u001b[39m\n\u001b[1;32m     53\u001b[0m         \u001b[38;5;124mf\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;130;01m\\n\u001b[39;00m\u001b[38;5;124m clients_total_loss_after :  \u001b[39m\u001b[38;5;132;01m{\u001b[39;00mfinal_test_result[\u001b[38;5;241m3\u001b[39m]\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m'\u001b[39m\n\u001b[1;32m     54\u001b[0m         )\n",
      "\u001b[0;31mTypeError\u001b[0m: federated_test() missing 1 required positional argument: 'lr'"
     ]
    }
   ],
   "source": [
    "if __name__ == \"__main__\":\n",
    "################################################# params ##################################################\n",
    "    num_clients = 10 # Specify the number of clients\n",
    "    frac = 10\n",
    "    global_epochs=100 #communication_round\n",
    "    local_epochs= 2\n",
    "    batch_size=32\n",
    "    lr=0.001\n",
    "    global_lr=1e-3\n",
    "    seed = 42\n",
    "    split_layer = 2\n",
    "    algo = \"pMixFed\" # 'FedSim' or FedAlt or 'FedAvg' or 'pFedMix'(older)  'pMixFed' or 'FedBABU' \n",
    "    n_hidden = 4 # number of hidden layer\n",
    "    mix_factor = 0.01\n",
    "    alpha = 0.5 # 0.5 or 1 # \n",
    "    beta = 0.1\n",
    "    dataset = \"CIFAR10\" # EMNIST , MNIST or CIFAR10 or CIFAR100\n",
    "    model = 'CNN' #RESNET18 or  #RESNET50 or #Mobile or #CNN\n",
    "    criterion = nn.CrossEntropyLoss() \n",
    "    ############################################### setup ###################################################\n",
    "\n",
    "    torch.cuda.empty_cache()\n",
    "    torch.backends.cudnn.benchmark = False\n",
    "\n",
    "    # torch.cuda.set_per_process_memory_fraction(1.0, device=0) \n",
    "\n",
    "    device = torch.device(\"cuda\") if torch.cuda.is_available() else torch.device(\"cpu\")\n",
    "\n",
    "    torch.manual_seed(seed)\n",
    "    torch.cuda.manual_seed(seed)\n",
    "    np.random.seed(seed)\n",
    "    random.seed(seed)\n",
    "    torch.backends.cudnn.benchmark = True\n",
    "    torch.backends.cudnn.deterministic = True\n",
    "\n",
    "    ############################################## Dataset #####################################################   \n",
    "    # Create non-IID datasets for each client\n",
    "    train_set, test_set = create_non_iid_datasets(num_clients, dataset, beta)\n",
    "    \n",
    "     ############################################## train ######################################################     \n",
    "    Final_train_loss , Final_train_accuracy , Final_test_accuracy = federated_train(dataset, algo, model , num_clients, global_epochs, batch_size, lr , global_lr, \n",
    "                                train_set, test_set, local_epochs, split_layer , mix_factor , alpha, beta, criterion , frac)\n",
    "\n",
    "  \n",
    " \n",
    "    \n",
    "     ############################################## test ###################################################### \n",
    "    \n",
    "    final_test_result = federated_test(algo, model, num_clients, dataset , test_set, criterion, local_epochs , beta)\n",
    "    print (f' clients_accuracy_before : {final_test_result[0]}'\n",
    "            f'\\n clients_total_loss_before: {final_test_result[1]}'\n",
    "            f'\\n clients_accuracy_after : {final_test_result[2]}'\n",
    "            f'\\n clients_total_loss_after :  {final_test_result[3]}'\n",
    "            )\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0b329e9b-e157-4d0d-9d04-468f5c433aae",
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_class_distribution(dataset, num_classes): \n",
    "    \"\"\"Computes the class distribution for a given dataset.\"\"\"\n",
    "    class_counts = torch.zeros(num_classes)\n",
    "    targets = [dataset.dataset.targets[idx] for idx in dataset.indices]\n",
    "    for t in targets:\n",
    "        class_counts[t] += 1\n",
    "    class_distribution = class_counts / len(targets)  # Normalize to get probabilities\n",
    "    return class_distribution.numpy()\n",
    "\n",
    "def estimate_alpha(client_datasets, num_classes):\n",
    "    \"\"\"Estimates the Dirichlet parameter alpha based on client class distributions.\"\"\"\n",
    "    # Collect all class distributions\n",
    "    class_distributions = [compute_class_distribution(dataset, num_classes) for dataset in client_datasets]\n",
    "\n",
    "    # Mean of class distributions\n",
    "    mean_distribution = np.mean(class_distributions, axis=0)\n",
    "\n",
    "    # Estimate alpha using method of moments\n",
    "    alpha = np.sum(mean_distribution) ** 2 / np.sum(mean_distribution ** 2)\n",
    "    \n",
    "    return alpha\n",
    "\n",
    "# Example usage\n",
    "# Assuming `client_datasets` is a list of datasets spliced using Dirichlet distribution\n",
    "num_classes = 100 # Change this to the number of classes in your dataset\n",
    "alpha_estimate = estimate_alpha(test_set, num_classes)\n",
    "print(f\"Estimated alpha for the Dirichlet distribution: {alpha_estimate:.4f}\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "67d392c5-ec40-4e90-8d00-04d3a37409b1",
   "metadata": {},
   "outputs": [],
   "source": [
    "with open(f'Results/resultsFedAvg-normal-CIFAR100-small_scale-Mobile-train.pkl', \"rb\") as file:\n",
    "    data = pickle.load(file)\n",
    "    Final_train_accuracy = data[\"Final_train_loss\"]\n",
    "\n",
    "\n",
    "print(Final_train_accuracy[42])\n",
    "# Plot the training loss\n",
    "plt.figure(figsize=(8, 6))\n",
    "rounds = np.arange(0,100) \n",
    "plt.plot( Final_train_accuracy, label='overall Training accuracy')\n",
    "    \n",
    "plt.xlabel('communication round')\n",
    "plt.ylabel('Accuracy')\n",
    "plt.title('overall Personalziation accuracy in each round of global training')\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "49c4fdc5-6725-4319-bdc3-6dbe47e8652b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "\n",
    "with open(f'Results/results', \"rb\") as file:\n",
    "    data = pickle.load(file)\n",
    "    Final_train_accuracy = data[\"clients_accuracy_before\"]\n",
    "\n",
    "\n",
    "print(Final_train_accuracy)\n",
    "# Plot the training loss\n",
    "# plt.figure(figsize=(8, 6))\n",
    "# rounds = np.arange(0,100) \n",
    "# plt.plot( Final_train_accuracy, label='overall Training accuracy')\n",
    "    \n",
    "# plt.xlabel('communication round')\n",
    "# plt.ylabel('Accuracy')\n",
    "# plt.title('overall Personalziation accuracy in each round of global training')\n",
    "# plt.legend()\n",
    "# plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d74c7483",
   "metadata": {},
   "outputs": [],
   "source": [
    "    final_test_result = federated_test(algo, model , num_clients, dataset , test_set, criterion, local_epochs)\n",
    "    print (f' clients_accuracy_before : {final_test_result[0]}'\n",
    "            f'\\n clients_total_loss_before: {final_test_result[1]}'\n",
    "            f'\\n clients_accuracy_after : {final_test_result[2]}'\n",
    "            f'\\n clients_total_loss_after :  {final_test_result[3]}'\n",
    "            )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "734dec1a-0f5f-4b50-b34e-08e0f6a2bebd",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "# Sample accuracy data for 10 clients\n",
    "clients = np.arange(1, 11)  # Client IDs 1 to 10\n",
    "accuracy = final_test_result[0].cpu()  # Sample accuracy values (mean 80, std dev 5)\n",
    "print(f'shape is : {accuracy.shape}')\n",
    "# Plotting accuracy per client\n",
    "plt.figure(figsize=(8, 6))\n",
    "\n",
    "# Scatter plot for accuracy\n",
    "plt.scatter(clients, accuracy, color='blue', s=100, alpha=0.7, label='Accuracy per client')\n",
    "\n",
    "# Plot mean accuracy\n",
    "mean_acc = np.mean(accuracy)\n",
    "plt.axhline(y=mean_acc, color='orange', linestyle='--', linewidth=2, label=f'Mean Accuracy: {mean_acc:.2f}')\n",
    "\n",
    "# Standard deviation range (shaded region)\n",
    "std_dev = np.std(accuracy)\n",
    "plt.fill_between(clients, mean_acc - std_dev, mean_acc + std_dev, color='gray', alpha=0.2, label=f'Std Dev: {std_dev:.2f}')\n",
    "\n",
    "# Customizing plot\n",
    "plt.title('Accuracy per Client with Mean and Std Dev', fontsize=16)\n",
    "plt.xlabel('Client ID', fontsize=14)\n",
    "plt.ylabel('Accuracy (%)', fontsize=14)\n",
    "plt.xticks(clients)\n",
    "plt.yticks(np.arange(min(accuracy)-5, max(accuracy)+5, 5))\n",
    "plt.grid(True)\n",
    "plt.legend()\n",
    "\n",
    "# Display plot\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3c9eacb2-e633-42c9-bfa7-2b45b3c95c17",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "import json\n",
    "\n",
    "with open(f'Results/resultsFedBABU-CIFAR10-small_scale-newCNN-train.pkl', 'rb') as file:\n",
    "    file1 = pickle.load(file)\n",
    "    results_FedBABU = [(100 - i) for i in file1[\"Final_train_accuracy\"]]\n",
    "    \n",
    "with open(f'Results/resultsFedAlt-CIFAR10-small_scale-train.pkl', 'rb') as file:\n",
    "    file2 = pickle.load(file)\n",
    "    results_FedAlt = [(100 - i) for i in file2[\"Final_train_accuracy\"]]\n",
    "\n",
    "with open(f'Results/resultsFedSim-CIFAR10-small_scale-train.pkl', 'rb') as file:\n",
    "    file2 = pickle.load(file)\n",
    "    results_FedSim = [(100 - i) for i in file2[\"Final_train_accuracy\"]]\n",
    "\n",
    "with open(f'Results/resultsFedAvg-CIFAR10-small_scale-train.pkl', 'rb') as file:\n",
    "    file2 = pickle.load(file)\n",
    "    results_FedAvg = [(100 - i) for i in file2[\"Final_train_accuracy\"]]\n",
    "\n",
    "\n",
    "with open(f'Results/resultspFedMix-CIFAR10-small_scale-train.pkl', 'rb') as file:\n",
    "    file3 = pickle.load(file)\n",
    "    results_pFedMix = [(100 - i) for i in file3[\"Final_train_accuracy\"]]\n",
    "\n",
    "\n",
    "with open(f'Results/resultspFedHN.pkl', 'rb') as file:\n",
    "    file4 = pickle.load(file)\n",
    "    results_pFedHN = [(100 - i) for i in file4[\"Final_train_accuracy\"]]\n",
    "    # results_pFedHN = [(100-i) for i in results_pFedHN1]\n",
    "    # print(f'size : {len(results_pFedHN)}')\n",
    "\n",
    "\n",
    "# with open(f'Results/results_2_inner_steps_seed_42.json', 'rb') as file:\n",
    "#     file3 = json.load(file)\n",
    "#     results_pFedHN1 = file3['test_avg_loss']\n",
    "#     results_pFedHN = [100-i for i in results_pFedHN1]\n",
    "#     print(f'size : {len(results_pFedHN)}')\n",
    "\n",
    "\n",
    "# Plotting the results\n",
    "plt.figure(figsize=(8, 6))\n",
    "rounds = np.arange(0,100) \n",
    "plt.plot(rounds, results_FedAvg, label='FedAvg', color='blue')\n",
    "plt.plot(rounds, results_FedAlt, label='FedAlt', color='green')\n",
    "plt.plot(rounds, results_FedSim, label='FedAlt', color='black')\n",
    "plt.plot(rounds, results_FedBABU, label='FedBABU', color='orange')\n",
    "plt.plot(rounds, results_pFedMix, label='pFedMix', color='red' )\n",
    "plt.plot(rounds, results_pFedHN, label='pFedHN', color='yellow')\n",
    "plt.xlabel('communication round')\n",
    "plt.ylabel('Performance')\n",
    "plt.title('Performance Comparison of different algorithms on  CIFAR10')\n",
    "plt.legend()\n",
    "plt.grid(True)\n",
    "plt.savefig('CIFAR10-train-accuracy.png')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ceea4da4-05c5-4ad9-9310-1da5691b5b12",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "import json\n",
    "\n",
    "with open(f'Results/resultsFedBABU-CIFAR10-small_scale-newCNN-train.pkl', 'rb') as file:\n",
    "    file1 = pickle.load(file)\n",
    "    results_FedBABU =  file1[\"Final_train_accuracy\"]\n",
    "    \n",
    "with open(f'Results/resultsFedAlt-CIFAR10-small_scale-train.pkl', 'rb') as file:\n",
    "    file2 = pickle.load(file)\n",
    "    results_FedAlt = file2[\"Final_train_accuracy\"]\n",
    "\n",
    "with open(f'Results/resultsFedSim-CIFAR10-small_scale-train.pkl', 'rb') as file:\n",
    "    file2 = pickle.load(file)\n",
    "    results_FedSim =  file2[\"Final_train_accuracy\"]\n",
    "\n",
    "with open(f'Results/resultsFedAvg-CIFAR10-small_scale-train.pkl', 'rb') as file:\n",
    "    file2 = pickle.load(file)\n",
    "    results_FedAvg =  file2[\"Final_train_accuracy\"]\n",
    "\n",
    "\n",
    "with open(f'Results/resultspFedMix-CIFAR10-small_scale-train.pkl', 'rb') as file:\n",
    "    file3 = pickle.load(file)\n",
    "    results_pFedMix = file3[\"Final_train_accuracy\"]\n",
    "\n",
    "\n",
    "with open(f'Results/resultspMixFed-CIFAR10-small_scale-CNN-train.pkl', 'rb') as file:\n",
    "    file3 = pickle.load(file)\n",
    "    results_pMixFed = file3[\"Final_train_accuracy\"]\n",
    "\n",
    "\n",
    "with open(f'Results/resultspFedHN-cifar10-10-train.pkl', 'rb') as file:\n",
    "    file4 = pickle.load(file)\n",
    "    results_pFedHN = [100 - i for i in file4[\"Final_train_loss\"]]\n",
    "    # results_pFedHN = [(100-i) for i in results_pFedHN1]\n",
    "    # print(f'size : {len(results_pFedHN)}')\n",
    "\n",
    "\n",
    "# with open(f'Results/results_2_inner_steps_seed_42.json', 'rb') as file:\n",
    "#     file3 = json.load(file)\n",
    "#     results_pFedHN1 = file3['test_avg_loss']\n",
    "#     results_pFedHN = [100-i for i in results_pFedHN1]\n",
    "#     print(f'size : {len(results_pFedHN)}')\n",
    "\n",
    "\n",
    "# Plotting the results\n",
    "plt.figure(figsize=(8, 6))\n",
    "rounds = np.arange(0,100) \n",
    "plt.plot(rounds, results_FedAvg, label='FedAvg', color='blue' , linewidth=2.5 )\n",
    "plt.plot(rounds, results_FedAlt, label='FedAlt', color='green' , linewidth=2.5 )\n",
    "plt.plot(rounds, results_FedSim, label='FedAlt', color='black' , linewidth=2.5 )\n",
    "plt.plot(rounds, results_FedBABU, label='FedBABU', color='orange' , linewidth=2.5 )\n",
    "plt.plot(rounds, results_pFedMix, label='pFedMix', color='red', linewidth=2.5  )\n",
    "# plt.plot(rounds, results_pMixFed, label='pMixFed', color='brown', linewidth=2.5  )\n",
    "plt.plot(rounds, results_pFedHN, label='pFedHN', color='yellow', linewidth=2.5 )\n",
    "plt.xlabel('communication round' , fontweight='bold' , fontsize=14)\n",
    "plt.ylabel('Performance' , fontweight='bold' , fontsize=14)\n",
    "plt.title('Performance Comparison of different algorithms on  CIFAR10' , fontweight='bold')\n",
    "plt.legend()\n",
    "plt.grid(True)\n",
    "plt.savefig('CIFAR10-train-accuracy.png')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "182f353b-b9ae-467c-8220-9e5e572fa0f8",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "with open(f'resultsFedAvg-MNIST.pkl', 'rb') as file:\n",
    "    results_FedAvg = pickle.load(file)\n",
    "    \n",
    "with open(f'resultsFedAlt-MNIST.pkl', 'rb') as file:\n",
    "    results_FedAlt = pickle.load(file)\n",
    "\n",
    "with open(f'resultspFedMix-MNIST.pkl', 'rb') as file:\n",
    "    results_pFedMix = pickle.load(file)\n",
    "\n",
    "\n",
    "# Plotting the results\n",
    "plt.figure(figsize=(8, 6))\n",
    "rounds = np.arange(0,100) \n",
    "plt.plot(rounds, results_FedAvg, label='FedAvg', color='blue')\n",
    "plt.plot(rounds, results_FedAlt, label='FedAlt', color='green')\n",
    "plt.plot(rounds, results_pFedMix, label='pFedMix', color='red')\n",
    "plt.xlabel('communication round')\n",
    "plt.ylabel('Performance')\n",
    "plt.title('Performance Comparison of different algorithms on MNIST')\n",
    "plt.legend()\n",
    "plt.grid(True)\n",
    "plt.savefig('sine_wave.png')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "88dd5bc9-1c5a-44cf-a46b-60b8f7103348",
   "metadata": {},
   "source": [
    "# ControlVersion\n",
    "\n",
    "### 18/1\n",
    "- For iterating over the data in the dataloader:\n",
    "  ```python\n",
    "  for x, y in dataloader:\n",
    "      x, y = x.to(device), y.to(device)\n",
    "      logit = model(x)\n",
    "      total_loss += criterion(logit, y)\n",
    "      pred = torch.softmax(logit, -1).argmax(-1)\n",
    "      acc += torch.eq(pred, y).int().sum()\n",
    "      num_samples += y.size(-1)\n",
    "Next step: Implement FedAVG (Averaging process) and client selection method (random or based on the \"have_seen\" attribute).\n",
    "Evaluation mode: Split the dataset into training, test, and evaluation sets for each communication round.\n",
    "### 03/27\n",
    "Finished implementing FedRecon, now fixing errors.\n",
    "Looking for a method to split the network with a cut layer \"k\" to separate global and local parameters.\n",
    "Fixed splitting process in models.\n",
    "Global and local parameters should be stored as OrderedDicts instead of lists.\n",
    "### 03/28\n",
    ". Reading and understanding mixup process, especially manifold mixup. (done)\n",
    "\n",
    "### 05/01\n",
    ". Implementing mixup (we still have cut layer) --> seeing other codes but use your own!\n",
    "\n",
    "### 05/13\n",
    "* the mixup is implemented in two stages with cut layer k. (note: pFedMe is not like FedAlt, it mixup the gm, lm and then do the training\n",
    "and then mixup between current global model and and lm(globam layers of local model) )\n",
    "\n",
    "### 05/14: \n",
    "mixup is implemented : both local_global_mixup and global_to_local_mixup and also mixup aggregation! :)))))))))))))))))\n",
    "\n",
    "\n",
    "\n",
    "### 05/15:\n",
    "* FedAlt is checked!\n",
    "*  the initialization of weights is checked --> it's pytorch default and it's costant as we use random seed ()\n",
    "*  checking pFedMix with positive (not negative, delta weight) --> done\n",
    "\n",
    "### 05/20:\n",
    "* MNIST is added --> the splitting non-i.i.d could be changed later -> [split](https://github.com/YasMinSdt/FedAlt/blob/master/data/partition/sort_and_partition.py#L2)\n",
    "\n",
    "### 07/11\"\n",
    "* check the global evaluation phase for FedAvg --> everything is correct\n",
    "\n",
    "### 07/19:\n",
    "* implementing global test for  \"FedAlt\", \"pFedMix\"\n",
    "  \n",
    "### 08/14:\n",
    "* running the test and train for CIFAR10\n",
    "\n",
    "### 09/09:\n",
    "* check and run the code for FedAVG , FedAlt\n",
    "* implemented FedSim\n",
    "\n",
    "## 09/12:\n",
    "* added CIFAR100\n",
    "* run pFedHN\n",
    "\n",
    "## 09/16:\n",
    "* FedBABU added\n",
    "* MobileNet added\n",
    "\n",
    "## Next step:\n",
    "* resnet18 / resnet 50\n",
    "* FedBABU / LG-FedAvg \n",
    "* adding visualisations --> look at the [it's here in the localhost](http://localhost:8010/notebooks/FL_partial_personalization/plotting/results1.ipynb) Line222 from this paper [partial PFL](https://arxiv.org/pdf/2204.03809)\n",
    "* Adding test (evaluation) phase for local training.\n",
    "* adding new data set. --> [github](dev/YasMinSdt/pFedHN/blob/main/experiments/pfedhn/trainer.py)\n",
    "* check the calculation of accuracy and loss ( we didn't use alpha_i or weighted averaging for test and train results)\n",
    "* change the NN architecture to resnet(from FedAlt code)\n",
    "\n",
    "\n",
    "## Concurrent runs:\n",
    "* goal 1 : running the test for unseen test data on clients on FedAvg/FedAlt/pFedMix with 100 clients on CIFAR10\n",
    "* goal 2: running the test for cold-start-users clients on FedAvg/FedAlt/pFedMix with 100 clients on CIFAR10 \n",
    "\n",
    "\n",
    "### Important tasks needs to be addressed! \n",
    "* Addressing different Models (ResNet, COnvnet , adaptaor , ..) --> [here](http://localhost:8010/edit/PFedHN/pFedHN/experiments/pfedhn/PFL_Mixup/models.py)\n",
    "* so we can have the local training way for fedalt or fedsim / or we can use just a simple finetunning (whichever gives us worst result!)\n",
    "* dataset split for CIFAR10 and CIFAR100 [load_data.py](http://localhost:8010/tree/PFedHN/pFedHN/experiments/pfedhn/PFL_Mixup)\n",
    "* Adding test (evaluation) phase for local training.\n",
    "* MNIST --> the splitting non-i.i.d could be changed later -> [split](https://github.com/YasMinSdt/FedAlt/blob/master/data/partition/sort_and_partition.py#L2)\n",
    "* dynamic layer wise mixup without cut layer using Mu\n",
    "* client selection in each communication round --> selected_clients = random.sample(train_clients, args.client_num_per_round)  //  for client_id in selected_clients:\n",
    "* Ensure that the global model is splitted according to the local model.\n",
    "* test and evaulation is before and after local training on test data --> we can implement in according to [split_dataset](https://github.com/YasMinSdt/FedAlt/blob/master/algorithms/fedavg.py#L90) where the data is splitted for each client\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1f9705c6-41c3-4ea4-9d23-5879bd98e9db",
   "metadata": {},
   "outputs": [],
   "source": [
    "## ControlVersion\n",
    "\n",
    "# ################################### 18/1 ####################################################\n",
    "#   for x, y in dataloader:\n",
    "#         x, y = x.to(device), y.to(device)\n",
    "#         logit = model(x)\n",
    "#         total_loss += criterion(logit, y)\n",
    "#         pred = torch.softmax(logit, -1).argmax(-1)\n",
    "#         acc += torch.eq(pred, y).int().sum()\n",
    "#         num_samples += y.size(-1)\n",
    "#     model.train()\n",
    "#     return total_loss, acc / num_samples\n",
    "\n",
    "\n",
    "# FedAVG --> Averaging process (aggregation) (done!)\n",
    "# selecting client ( we should pick a selection method) __> it could be random () the have_seen attribiute \n",
    "# is refering to this\n",
    "# evaluation mode\n",
    "# dataset\n",
    "# for each communication round split the dataset to training, test and evaluation\n",
    "#\n",
    "\n",
    "# \n",
    "\n",
    "#   next step : partial global / local model (parameter decoupling ) --> you can just get insights from FedRecon + pytorch \n",
    "\n",
    "# working on the model (for making it split ) ---> https://github.com/KarhouTam/FedRecon/blob/e244de7725d17be9d06285d0687a1680f6d0821d/model.py#L14\n",
    "# L109 split_model ---> done! --> ( later you can extend it with PFL facebook code in models/ split_server...)\n",
    "\n",
    "#  I realized we can still do mixup for fedAlt, since we can do mixup only for\n",
    "#the shared layers, however it's best to use it for fedsim! \n",
    "\n",
    "############################# 03/27 ############################################\n",
    "# finished fedrecon! just start fixing the errors :) good luck smartii  --> fixed it !\n",
    "# looking for a split that with a cut layer k could split the network \n",
    "## fixing the spliting process L16 in models here -->  // check the fedrecon code , Fb code was no good --> done!\n",
    "# the global and local param should be a orderd dict {} and not a list (to ba able to load in state_disct) --> done!\n",
    "# global params not in order , local is not working! --> check for not any or anything --> done\n",
    "\n",
    "############################# 03/28 ############################################\n",
    "# reading and underestanding mixup process (espesially manifold mixup)\n",
    "# mixup implementation\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "# adding test (evaluation) phase for local training ()\n",
    "# important : the global model should be splitted according to the local model!"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.9.0"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
