{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "vgUXRKpJh1yE"
   },
   "source": [
    "# FedChill"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "execution": {
     "iopub.execute_input": "2025-09-16T15:34:43.127662Z",
     "iopub.status.busy": "2025-09-16T15:34:43.127376Z",
     "iopub.status.idle": "2025-09-16T15:35:00.431464Z",
     "shell.execute_reply": "2025-09-16T15:35:00.430814Z",
     "shell.execute_reply.started": "2025-09-16T15:34:43.127640Z"
    },
    "id": "_dq_ZRdAh1yG",
    "outputId": "f166c1fd-2eb3-4852-95e5-244884bbb5d8",
    "trusted": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Using device: cuda\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "import torch.nn.functional as F\n",
    "import torchvision\n",
    "import torchvision.transforms as transforms\n",
    "import numpy as np\n",
    "from torch.utils.data import Dataset, DataLoader, Subset, random_split\n",
    "import matplotlib.pyplot as plt\n",
    "from collections import Counter\n",
    "import copy\n",
    "import random\n",
    "from typing import Dict, List, Tuple\n",
    "import math\n",
    "\n",
    "import pandas as pd\n",
    "\n",
    "torch.manual_seed(42)\n",
    "np.random.seed(42)\n",
    "random.seed(42)\n",
    "\n",
    "device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "print(f\"Using device: {device}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "2NKqWJJTS7GB"
   },
   "source": [
    "---"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T15:35:06.356206Z",
     "iopub.status.busy": "2025-09-16T15:35:06.355482Z",
     "iopub.status.idle": "2025-09-16T15:35:06.359531Z",
     "shell.execute_reply": "2025-09-16T15:35:06.358932Z",
     "shell.execute_reply.started": "2025-09-16T15:35:06.356180Z"
    },
    "trusted": true
   },
   "outputs": [],
   "source": [
    "NUM_CLASSES = 10"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T15:35:08.614783Z",
     "iopub.status.busy": "2025-09-16T15:35:08.614089Z",
     "iopub.status.idle": "2025-09-16T15:35:08.626462Z",
     "shell.execute_reply": "2025-09-16T15:35:08.625640Z",
     "shell.execute_reply.started": "2025-09-16T15:35:08.614759Z"
    },
    "id": "gVCZKSAhh1yH",
    "trusted": true
   },
   "outputs": [],
   "source": [
    "class SimpleCNN(nn.Module):\n",
    "    def __init__(self, num_classes=NUM_CLASSES):\n",
    "        super(SimpleCNN, self).__init__()\n",
    "\n",
    "        \n",
    "        self.conv1 = nn.Conv2d(3, 64, 3, padding=1)\n",
    "        self.bn1 = nn.BatchNorm2d(64)\n",
    "        self.conv2 = nn.Conv2d(64, 64, 3, padding=1)\n",
    "        self.bn2 = nn.BatchNorm2d(64)\n",
    "\n",
    "        \n",
    "        self.conv3 = nn.Conv2d(64, 128, 3, padding=1)\n",
    "        self.bn3 = nn.BatchNorm2d(128)\n",
    "        self.conv4 = nn.Conv2d(128, 128, 3, padding=1)\n",
    "        self.bn4 = nn.BatchNorm2d(128)\n",
    "\n",
    "        \n",
    "        self.conv5 = nn.Conv2d(128, 256, 3, padding=1)\n",
    "        self.bn5 = nn.BatchNorm2d(256)\n",
    "        self.conv6 = nn.Conv2d(256, 256, 3, padding=1)\n",
    "        self.bn6 = nn.BatchNorm2d(256)\n",
    "\n",
    "        \n",
    "        self.conv7 = nn.Conv2d(256, 512, 3, padding=1)\n",
    "        self.bn7 = nn.BatchNorm2d(512)\n",
    "\n",
    "        self.pool = nn.MaxPool2d(2, 2)\n",
    "        self.adaptive_pool = nn.AdaptiveAvgPool2d((2, 2))  \n",
    "\n",
    "        \n",
    "        self.fc1 = nn.Linear(512 * 2 * 2, 1024)\n",
    "        self.bn_fc1 = nn.BatchNorm1d(1024)\n",
    "        self.dropout1 = nn.Dropout(0.5)\n",
    "\n",
    "        self.fc2 = nn.Linear(1024, 512)\n",
    "        self.bn_fc2 = nn.BatchNorm1d(512)\n",
    "        self.dropout2 = nn.Dropout(0.3)\n",
    "\n",
    "        self.fc3 = nn.Linear(512, num_classes)\n",
    "\n",
    "    def forward(self, x):\n",
    "        \n",
    "        x = F.relu(self.bn1(self.conv1(x)))\n",
    "        x = F.relu(self.bn2(self.conv2(x)))\n",
    "        x = self.pool(x)  \n",
    "\n",
    "        \n",
    "        x = F.relu(self.bn3(self.conv3(x)))\n",
    "        x = F.relu(self.bn4(self.conv4(x)))\n",
    "        x = self.pool(x)  \n",
    "\n",
    "        \n",
    "        x = F.relu(self.bn5(self.conv5(x)))\n",
    "        x = F.relu(self.bn6(self.conv6(x)))\n",
    "        x = self.pool(x)  \n",
    "\n",
    "        \n",
    "        x = F.relu(self.bn7(self.conv7(x)))\n",
    "        x = self.adaptive_pool(x)  \n",
    "\n",
    "        \n",
    "        x = x.view(x.size(0), -1)  \n",
    "\n",
    "        x = F.relu(self.bn_fc1(self.fc1(x)))\n",
    "        x = self.dropout1(x)\n",
    "\n",
    "        x = F.relu(self.bn_fc2(self.fc2(x)))\n",
    "        x = self.dropout2(x)\n",
    "\n",
    "        x = self.fc3(x)\n",
    "        return x\n",
    "\n",
    "    def get_features(self, x):\n",
    "        \n",
    "        x = F.relu(self.bn1(self.conv1(x)))\n",
    "        x = F.relu(self.bn2(self.conv2(x)))\n",
    "        x = self.pool(x)\n",
    "\n",
    "        \n",
    "        x = F.relu(self.bn3(self.conv3(x)))\n",
    "        x = F.relu(self.bn4(self.conv4(x)))\n",
    "        x = self.pool(x)\n",
    "\n",
    "        \n",
    "        x = F.relu(self.bn5(self.conv5(x)))\n",
    "        x = F.relu(self.bn6(self.conv6(x)))\n",
    "        x = self.pool(x)\n",
    "\n",
    "        \n",
    "        x = F.relu(self.bn7(self.conv7(x)))\n",
    "        x = self.adaptive_pool(x)\n",
    "\n",
    "        \n",
    "        x = x.view(x.size(0), -1)\n",
    "        x = F.relu(self.bn_fc1(self.fc1(x)))\n",
    "        x = self.dropout1(x)\n",
    "        x = F.relu(self.bn_fc2(self.fc2(x)))\n",
    "\n",
    "        return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "execution_failed": "2025-09-16T09:12:12.991Z"
    },
    "trusted": true
   },
   "outputs": [],
   "source": [
    "from torchsummary import summary\n",
    "\n",
    "NUM_CLASSES = 10  \n",
    "\n",
    "\n",
    "model = SimpleCNN(num_classes=NUM_CLASSES).to(device)\n",
    "\n",
    "summary(model, (3, 32, 32))  "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T15:35:13.559735Z",
     "iopub.status.busy": "2025-09-16T15:35:13.559197Z",
     "iopub.status.idle": "2025-09-16T15:35:13.563701Z",
     "shell.execute_reply": "2025-09-16T15:35:13.563136Z",
     "shell.execute_reply.started": "2025-09-16T15:35:13.559709Z"
    },
    "trusted": true
   },
   "outputs": [],
   "source": [
    "\n",
    "BATCH_SIZE = 64\n",
    "LEARNING_RATE = 0.01\n",
    "LOCAL_EPOCHS = 5\n",
    "NUM_OF_CLIENTS = 10\n",
    "COMM_ROUND = 50\n",
    "ALPHA = 0.5  \n",
    "FRAC = 0.1   "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T15:35:15.677581Z",
     "iopub.status.busy": "2025-09-16T15:35:15.676996Z",
     "iopub.status.idle": "2025-09-16T15:35:15.692808Z",
     "shell.execute_reply": "2025-09-16T15:35:15.692233Z",
     "shell.execute_reply.started": "2025-09-16T15:35:15.677555Z"
    },
    "id": "CImsSomMh1yH",
    "trusted": true
   },
   "outputs": [],
   "source": [
    "def load_and_partition_data(num_clients=NUM_OF_CLIENTS, alpha=ALPHA, batch_size=BATCH_SIZE, frac=FRAC, rand_seed=42):\n",
    "    \n",
    "    transform = transforms.Compose([\n",
    "        transforms.ToTensor(),\n",
    "        transforms.Normalize((0.5, 0.5, 0.5), (0.5, 0.5, 0.5))\n",
    "    ])\n",
    "\n",
    "    \n",
    "    torch.manual_seed(rand_seed)\n",
    "    np.random.seed(rand_seed)\n",
    "\n",
    "    \n",
    "    full_dataset = torchvision.datasets.CIFAR10(root='./data', train=True, download=True, transform=transform)\n",
    "    test_dataset = torchvision.datasets.CIFAR10(root='./data', train=False, download=True, transform=transform)\n",
    "\n",
    "    \n",
    "    y_train = np.array(full_dataset.targets)\n",
    "    y_test = np.array(test_dataset.targets)\n",
    "\n",
    "    num_classes = 10\n",
    "    N = len(full_dataset)  \n",
    "    N_test = len(test_dataset)  \n",
    "\n",
    "    \n",
    "    net_dataidx_map = {}\n",
    "    net_dataidx_map_test = {}\n",
    "\n",
    "    \n",
    "    min_size = 0\n",
    "    while min_size < 10:  \n",
    "        idx_batch = [[] for _ in range(num_clients)]\n",
    "        idx_batch_test = [[] for _ in range(num_clients)]\n",
    "        for k in range(num_classes):\n",
    "            idx_k = np.where(y_train == k)[0]\n",
    "            idx_k_test = np.where(y_test == k)[0]\n",
    "            np.random.shuffle(idx_k)  \n",
    "            np.random.shuffle(idx_k_test)  \n",
    "            \n",
    "            proportions = np.random.dirichlet(np.repeat(alpha, num_clients))\n",
    "            \n",
    "            proportions_train = np.array([p * (len(idx_j) < N / num_clients) for p, idx_j in zip(proportions, idx_batch)])\n",
    "            proportions_test = np.array([p * (len(idx_j) < N_test / num_clients) for p, idx_j in zip(proportions, idx_batch_test)])\n",
    "            proportions_train = proportions_train / proportions_train.sum()  \n",
    "            proportions_test = proportions_test / proportions_test.sum()  \n",
    "            proportions_train = (np.cumsum(proportions_train) * len(idx_k)).astype(int)[:-1]\n",
    "            proportions_test = (np.cumsum(proportions_test) * len(idx_k_test)).astype(int)[:-1]\n",
    "            \n",
    "            idx_batch = [idx_j + idx.tolist() for idx_j, idx in zip(idx_batch, np.split(idx_k, proportions_train))]\n",
    "            idx_batch_test = [idx_j + idx.tolist() for idx_j, idx in zip(idx_batch_test, np.split(idx_k_test, proportions_test))]\n",
    "        min_size = min([len(idx_j) for idx_j in idx_batch])  \n",
    "\n",
    "    \n",
    "    for j in range(num_clients):\n",
    "        np.random.shuffle(idx_batch[j])\n",
    "        np.random.shuffle(idx_batch_test[j])\n",
    "        net_dataidx_map[j] = idx_batch[j]\n",
    "        net_dataidx_map_test[j] = idx_batch_test[j]\n",
    "\n",
    "    \n",
    "    client_train_loaders = []\n",
    "    client_val_loaders = []\n",
    "    client_class_distributions = []\n",
    "\n",
    "    for i in range(num_clients):\n",
    "        \n",
    "        np.random.seed(rand_seed + i)\n",
    "\n",
    "        \n",
    "        num_data = len(net_dataidx_map[i])\n",
    "        frac_num_data = int(frac * num_data)\n",
    "        frac_indices = np.random.choice(num_data, frac_num_data, replace=False)\n",
    "        train_indices = [net_dataidx_map[i][j] for j in frac_indices]\n",
    "\n",
    "        \n",
    "        num_data_test = len(net_dataidx_map_test[i])\n",
    "        frac_num_data_test = int(min(2 * frac, 1.0) * num_data_test)  \n",
    "        frac_indices_test = np.random.choice(num_data_test, frac_num_data_test, replace=False)\n",
    "        val_indices = [net_dataidx_map_test[i][j] for j in frac_indices_test]\n",
    "\n",
    "        \n",
    "        client_labels = [y_train[idx] for idx in train_indices]\n",
    "        class_counts = Counter(client_labels)\n",
    "        distribution = {cls: class_counts.get(cls, 0) / len(client_labels) if len(client_labels) > 0 else 0 for cls in range(num_classes)}\n",
    "        client_class_distributions.append(distribution)\n",
    "\n",
    "        \n",
    "        client_train_dataset = Subset(full_dataset, train_indices)\n",
    "        client_val_dataset = Subset(test_dataset, val_indices)\n",
    "\n",
    "        \n",
    "        g_train = torch.Generator().manual_seed(rand_seed + i)\n",
    "        g_val = torch.Generator().manual_seed(rand_seed + i + num_clients)\n",
    "\n",
    "        train_loader = DataLoader(client_train_dataset, batch_size=batch_size,\n",
    "                                 shuffle=True, generator=g_train, drop_last=True)\n",
    "        val_loader = DataLoader(client_val_dataset, batch_size=batch_size,\n",
    "                                shuffle=True, generator=g_val, drop_last=True)\n",
    "\n",
    "        client_train_loaders.append(train_loader)\n",
    "        client_val_loaders.append(val_loader)\n",
    "\n",
    "    \n",
    "    g_test = torch.Generator().manual_seed(rand_seed + 2 * num_clients + 1)\n",
    "    test_loader = DataLoader(test_dataset, batch_size=batch_size, shuffle=True,\n",
    "                             generator=g_test, num_workers=2)\n",
    "\n",
    "    print(\"Data partitioning complete.\")\n",
    "\n",
    "    \n",
    "    torch.manual_seed(rand_seed)\n",
    "    np.random.seed(rand_seed)\n",
    "\n",
    "    return client_train_loaders, client_val_loaders, test_loader, client_class_distributions"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "oJrL1U1Hh1yJ"
   },
   "source": [
    "---"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "OOr1Qsu_S7GC"
   },
   "source": [
    "## Utility Functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T15:35:20.557027Z",
     "iopub.status.busy": "2025-09-16T15:35:20.556413Z",
     "iopub.status.idle": "2025-09-16T15:35:20.567267Z",
     "shell.execute_reply": "2025-09-16T15:35:20.566621Z",
     "shell.execute_reply.started": "2025-09-16T15:35:20.557004Z"
    },
    "trusted": true
   },
   "outputs": [],
   "source": [
    "class AdaptiveClientHyperparams:\n",
    "    def __init__(self, initial_lr=0.01, min_lr=0.0001, max_lr=0.05, \n",
    "                 initial_temp=0.5, min_temp=0.05, max_temp=1.0,\n",
    "                 patience=2, lr_decay_factor=0.7, temp_adjust_factor=0.8):\n",
    "        \n",
    "        self.lr = initial_lr\n",
    "        self.min_lr = min_lr\n",
    "        self.max_lr = max_lr\n",
    "        self.lr_decay_factor = lr_decay_factor\n",
    "        \n",
    "        \n",
    "        self.temp = initial_temp\n",
    "        self.min_temp = min_temp\n",
    "        self.max_temp = max_temp\n",
    "        self.temp_adjust_factor = temp_adjust_factor\n",
    "        \n",
    "        \n",
    "        self.patience = patience\n",
    "        self.performance_history = []\n",
    "        self.stagnation_count = 0\n",
    "        self.improvement_count = 0\n",
    "        self.het_score = 0.0    \n",
    "        \n",
    "    def update_temp_from_heterogeneity(self, het_score):\n",
    "        \"\"\"Initialize temperature based on client's heterogeneity score\"\"\"\n",
    "        self.het_score = het_score\n",
    "        \n",
    "        \n",
    "        self.temp = self.max_temp * math.exp(-3 * het_score) \n",
    "        self.temp = min(max(self.temp, self.min_temp), self.max_temp)\n",
    "        \n",
    "    def record_performance(self, accuracy):\n",
    "        \"\"\"Record performance and adjust hyperparameters if needed\"\"\"\n",
    "        self.performance_history.append(accuracy)\n",
    "        \n",
    "        \n",
    "        if len(self.performance_history) >= 3:\n",
    "            \n",
    "            if self.performance_history[-1] <= self.performance_history[-2]:\n",
    "                self.stagnation_count += 1\n",
    "                self.improvement_count = 0\n",
    "            else:\n",
    "                self.improvement_count += 1\n",
    "                self.stagnation_count = 0\n",
    "                \n",
    "            \n",
    "            if self.stagnation_count >= self.patience:\n",
    "                \n",
    "                if self.temp > self.min_temp * 1.1:  \n",
    "                    self.temp *= self.temp_adjust_factor\n",
    "                    self.temp = max(self.temp, self.min_temp)\n",
    "                    self.stagnation_count = 0\n",
    "                    return True, \"decrease-temp\", self.temp\n",
    "        \n",
    "        return False, \"unchanged\", None\n",
    "        \n",
    "    def get_lr(self):\n",
    "        return self.lr\n",
    "        \n",
    "    def get_temp(self):\n",
    "        return self.temp\n",
    "    \n",
    "\n",
    "def calculate_heterogeneity_score(client_distribution, global_distribution):\n",
    "    \"\"\"Calculate how different client's class distribution is from global distribution\"\"\"\n",
    "    score = 0\n",
    "    for cls in range(10):\n",
    "        client_prob = client_distribution.get(cls, 0)\n",
    "        global_prob = global_distribution.get(cls, 1/10)\n",
    "        if client_prob > 0 and global_prob > 0:\n",
    "            ratio = client_prob / global_prob\n",
    "            score += abs(ratio - 1)\n",
    "    \n",
    "    \n",
    "    score = min(score / 10, 1)\n",
    "    return score\n",
    "\n",
    "def calculate_global_distribution(client_class_distributions):\n",
    "    \"\"\"Calculate approximate global class distribution from client distributions\"\"\"\n",
    "    global_dist = {}\n",
    "    n_clients = len(client_class_distributions)\n",
    "    \n",
    "    for cls in range(10):\n",
    "        global_dist[cls] = sum(dist.get(cls, 0) for dist in client_class_distributions) / n_clients\n",
    "    \n",
    "    for cls in range(10):\n",
    "        if global_dist[cls] < 0.01:\n",
    "            global_dist[cls] = 0.01\n",
    "    \n",
    "    total = sum(global_dist.values())\n",
    "    global_dist = {k: v/total for k, v in global_dist.items()}\n",
    "    \n",
    "    return global_dist"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "---"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T15:35:47.066717Z",
     "iopub.status.busy": "2025-09-16T15:35:47.066236Z",
     "iopub.status.idle": "2025-09-16T15:35:47.077863Z",
     "shell.execute_reply": "2025-09-16T15:35:47.077279Z",
     "shell.execute_reply.started": "2025-09-16T15:35:47.066692Z"
    },
    "id": "f2GMhq3Qh1yJ",
    "trusted": true
   },
   "outputs": [],
   "source": [
    "def client_train(model, train_loader, optimizer, epochs, print_flag=True):\n",
    "    model.train()\n",
    "    for epoch in range(epochs):\n",
    "        running_loss = 0.0\n",
    "        for inputs, targets in train_loader:\n",
    "            inputs, targets = inputs.to(device), targets.to(device)\n",
    "            optimizer.zero_grad()\n",
    "            outputs = model(inputs)\n",
    "            loss = F.cross_entropy(outputs, targets)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            running_loss += loss.item()\n",
    "\n",
    "        if print_flag:\n",
    "            print(f\"Epoch [{epoch+1}/{epochs}], Loss: {running_loss/len(train_loader):.4f}\")\n",
    "\n",
    "def aggregate_models(global_model, client_models, client_indices, train_loaders):\n",
    "    \n",
    "    global_dict = global_model.state_dict()\n",
    "    \n",
    "    \n",
    "    n_samples = [len(train_loaders[idx].dataset) for idx in client_indices]\n",
    "    total_samples = sum(n_samples)\n",
    "    \n",
    "    \n",
    "    aggregated_dict = {}\n",
    "    \n",
    "    \n",
    "    for key in global_dict.keys():\n",
    "        aggregated_dict[key] = torch.zeros_like(global_dict[key], dtype=torch.float32)\n",
    "    \n",
    "    \n",
    "    for i, idx in enumerate(client_indices):\n",
    "        client_dict = client_models[idx].state_dict()\n",
    "        weight = n_samples[i] / total_samples\n",
    "        \n",
    "        for key in global_dict.keys():\n",
    "            \n",
    "            aggregated_dict[key] += client_dict[key].float() * weight\n",
    "    \n",
    "    \n",
    "    for key in global_dict.keys():\n",
    "        \n",
    "        global_dict[key] = aggregated_dict[key].to(dtype=global_dict[key].dtype)\n",
    "    \n",
    "    global_model.load_state_dict(global_dict)\n",
    "\n",
    "\n",
    "def evaluate_model(model, data_loader):\n",
    "\n",
    "    model.eval()\n",
    "    correct = 0\n",
    "    total = 0\n",
    "    with torch.no_grad():\n",
    "        for inputs, targets in data_loader:\n",
    "            inputs, targets = inputs.to(device), targets.to(device)\n",
    "            outputs = model(inputs)\n",
    "            _, predicted = torch.max(outputs, 1)\n",
    "            total += targets.size(0)\n",
    "            correct += (predicted == targets).sum().item()\n",
    "    return 100 * correct / total\n",
    "\n",
    "def client_train_with_temp(model, train_loader, optimizer, epochs, temperature=1.0, print_flag=True):\n",
    "    model.train()\n",
    "    for epoch in range(epochs):\n",
    "        running_loss = 0.0\n",
    "        for inputs, targets in train_loader:\n",
    "            inputs, targets = inputs.to(device), targets.to(device)\n",
    "            optimizer.zero_grad()\n",
    "            outputs = model(inputs)\n",
    "\n",
    "            \n",
    "            if temperature != 1.0:\n",
    "                \n",
    "                log_probs = F.log_softmax(outputs / temperature, dim=1)\n",
    "                loss = F.nll_loss(log_probs, targets)\n",
    "            else:\n",
    "                \n",
    "                loss = F.cross_entropy(outputs, targets)\n",
    "\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            running_loss += loss.item()\n",
    "\n",
    "        if print_flag:\n",
    "            print(f\"Epoch [{epoch+1}/{epochs}], Loss: {running_loss/len(train_loader):.4f}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "_ybsTIJmS7GD"
   },
   "source": [
    "---"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T09:14:25.445015Z",
     "iopub.status.busy": "2025-09-16T09:14:25.444749Z",
     "iopub.status.idle": "2025-09-16T09:14:25.462468Z",
     "shell.execute_reply": "2025-09-16T09:14:25.461820Z",
     "shell.execute_reply.started": "2025-09-16T09:14:25.444995Z"
    },
    "trusted": true
   },
   "outputs": [],
   "source": [
    "def run_fedchill():\n",
    "    print_flag = False\n",
    "\n",
    "    client_train_loaders, client_val_loaders, test_loader, client_class_distributions = load_and_partition_data(\n",
    "        num_clients=NUM_OF_CLIENTS, \n",
    "        alpha=ALPHA, \n",
    "        batch_size=BATCH_SIZE, \n",
    "        frac=FRAC, \n",
    "        rand_seed=42\n",
    "    )\n",
    "    num_clients = len(client_train_loaders)\n",
    "    num_clients = len(client_train_loaders)\n",
    "\n",
    "    \n",
    "    global_distribution = calculate_global_distribution(client_class_distributions)\n",
    "    \n",
    "    \n",
    "    client_params = [AdaptiveClientHyperparams() for _ in range(num_clients)]\n",
    "    \n",
    "    \n",
    "    for i, distribution in enumerate(client_class_distributions):\n",
    "        het_score = calculate_heterogeneity_score(distribution, global_distribution)\n",
    "        client_params[i].update_temp_from_heterogeneity(het_score)\n",
    "        print(f\"Client {i} - Het Score: {het_score:.4f}, Initial temp: {client_params[i].get_temp():.4f}\")\n",
    "\n",
    "    global_model = SimpleCNN().to(device)\n",
    "    round_server_acc_list = []\n",
    "    round_client_acc_list = [[] for _ in range(num_clients)]\n",
    "    \n",
    "    \n",
    "    client_temp_history = [[] for _ in range(num_clients)]\n",
    "    client_lr_history = [[] for _ in range(num_clients)]\n",
    "\n",
    "    client_models = [copy.deepcopy(global_model).to(device) for _ in range(num_clients)]\n",
    "    client_optimizers = [optim.SGD(model.parameters(), lr=client_params[i].get_lr()) for i, model in enumerate(client_models)]\n",
    "\n",
    "    for comm_round in range(COMM_ROUND):\n",
    "        print(f\"\\n{'='*20} COMMUNICATION ROUND {comm_round+1} {'='*20}\")\n",
    "\n",
    "        \n",
    "        selected_clients = list(range(num_clients))\n",
    "\n",
    "        \n",
    "        if comm_round == COMM_ROUND - 1:\n",
    "            print_flag = True\n",
    "            print(\"\\n--- LOCAL TRAINING OF CLIENTS ---\")\n",
    "\n",
    "        for client_idx in selected_clients:\n",
    "            \n",
    "            client_models[client_idx].load_state_dict(global_model.state_dict())\n",
    "            \n",
    "            \n",
    "            curr_lr = client_params[client_idx].get_lr()\n",
    "            curr_temp = client_params[client_idx].get_temp()\n",
    "            \n",
    "            \n",
    "            client_temp_history[client_idx].append(curr_temp)\n",
    "            client_lr_history[client_idx].append(curr_lr)\n",
    "            \n",
    "            \n",
    "            client_optimizers[client_idx] = optim.SGD(client_models[client_idx].parameters(), lr=curr_lr)\n",
    "\n",
    "            if comm_round == COMM_ROUND - 1:\n",
    "                print(f\"\\n--- Client {client_idx} Local Training with T={curr_temp:.4f}, LR={curr_lr:.6f} ---\")\n",
    "\n",
    "            \n",
    "            client_train_with_temp(\n",
    "                client_models[client_idx],\n",
    "                client_train_loaders[client_idx],\n",
    "                client_optimizers[client_idx],\n",
    "                LOCAL_EPOCHS,\n",
    "                temperature=curr_temp,\n",
    "                print_flag=print_flag\n",
    "            )\n",
    "\n",
    "            \n",
    "            local_acc = evaluate_model(client_models[client_idx], client_val_loaders[client_idx])\n",
    "            if comm_round == COMM_ROUND - 1:\n",
    "                print(f\"Client {client_idx} Private Data Validation Accuracy: {local_acc:.2f}%\")\n",
    "\n",
    "            \n",
    "            client_testset_acc = evaluate_model(client_models[client_idx], test_loader)\n",
    "            round_client_acc_list[client_idx].append(client_testset_acc)\n",
    "            \n",
    "            \n",
    "            adjusted, param_type, new_value = client_params[client_idx].record_performance(local_acc)\n",
    "            if adjusted and (comm_round == COMM_ROUND - 1 or print_flag):\n",
    "                print(f\"Client {client_idx} - {param_type} to {new_value:.6f}\")\n",
    "\n",
    "        \n",
    "        aggregate_models(global_model, client_models, selected_clients, client_train_loaders)\n",
    "\n",
    "        \n",
    "        server_test_val_acc = evaluate_model(global_model, test_loader)\n",
    "        if comm_round == COMM_ROUND - 1:\n",
    "            print(f\"\\n--- Server Evaluation ---\")\n",
    "            print(f\"Server Test Data Validation Accuracy: {server_test_val_acc:.2f}%\")\n",
    "\n",
    "        round_server_acc_list.append(server_test_val_acc)\n",
    "\n",
    "    \n",
    "    rounds = list(range(1, COMM_ROUND + 1))\n",
    "    plt.figure(figsize=(8, 6))\n",
    "    plt.plot(rounds, round_server_acc_list, marker='o', linewidth=2, label='Server Accuracy')\n",
    "    plt.xlabel('Communication Rounds')\n",
    "    plt.ylabel('Accuracy (%)')\n",
    "    plt.title('Server Accuracy over Communication Rounds')\n",
    "    plt.legend()\n",
    "    plt.grid(True)\n",
    "    plt.show()\n",
    "\n",
    "    \n",
    "    plt.figure(figsize=(10, 6))\n",
    "    for i in range(num_clients):\n",
    "        plt.plot(rounds, round_client_acc_list[i], marker='o', label=f'Client {i}')\n",
    "    plt.xlabel('Communication Rounds')\n",
    "    plt.ylabel('Accuracy (%)')\n",
    "    plt.title('Client Test Accuracy over Communication Rounds')\n",
    "    plt.legend()\n",
    "    plt.grid(True)\n",
    "    plt.show()\n",
    "    \n",
    "    \n",
    "    plt.figure(figsize=(10, 6))\n",
    "    for i in range(num_clients):\n",
    "        plt.plot(rounds, client_temp_history[i], marker='s', label=f'Client {i}')\n",
    "    plt.xlabel('Communication Rounds')\n",
    "    plt.ylabel('Temperature')\n",
    "    plt.title('Client Temperature Adaptation over Rounds')\n",
    "    plt.legend()\n",
    "    plt.grid(True)\n",
    "    plt.show()\n",
    "    \n",
    "    \n",
    "    plt.figure(figsize=(10, 6))\n",
    "    for i in range(num_clients):\n",
    "        plt.plot(rounds, client_lr_history[i], marker='d', label=f'Client {i}')\n",
    "    plt.xlabel('Communication Rounds')\n",
    "    plt.ylabel('Learning Rate')\n",
    "    plt.title('Client Learning Rate Adaptation over Rounds')\n",
    "    plt.legend()\n",
    "    plt.grid(True)\n",
    "    plt.show()\n",
    "\n",
    "    \n",
    "    print(\"\\n--- Clients Final Evaluation on Private Data sets ---\")\n",
    "    for i in range(num_clients):\n",
    "        local_acc = evaluate_model(client_models[i], client_val_loaders[i])\n",
    "        print(f\"Client {i} Final Private Data Validation Accuracy: {local_acc:.2f}%\")\n",
    "\n",
    "    final_test_acc = evaluate_model(global_model, test_loader)\n",
    "    print(f\"\\nFinal Server Test Accuracy: {final_test_acc:.2f}%\")\n",
    "\n",
    "    return round_server_acc_list, round_client_acc_list, client_temp_history, client_lr_history, global_model"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "---"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T09:14:27.572902Z",
     "iopub.status.busy": "2025-09-16T09:14:27.572543Z",
     "iopub.status.idle": "2025-09-16T09:14:31.365137Z",
     "shell.execute_reply": "2025-09-16T09:14:31.364195Z",
     "shell.execute_reply.started": "2025-09-16T09:14:27.572875Z"
    },
    "trusted": true
   },
   "outputs": [],
   "source": [
    "\n",
    "BATCH_SIZE = 64\n",
    "LEARNING_RATE = 0.01\n",
    "LOCAL_EPOCHS = 5\n",
    "NUM_OF_CLIENTS = 10\n",
    "COMM_ROUND = 50\n",
    "ALPHA = 0.5  \n",
    "FRAC = 0.1   \n",
    "\n",
    "\n",
    "print(f\"\\n\\n{'#'*50}\")\n",
    "print(f\"# RUNNING FEDCHILL\")\n",
    "print(f\"{'#'*50}\\n\")\n",
    "\n",
    "server_acc, client_acc, temp_history, lr_history, final_model = run_fedchill()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Running Complete FedChill across Different Measures of Heterogeneity"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T16:11:49.268834Z",
     "iopub.status.busy": "2025-09-16T16:11:49.267998Z",
     "iopub.status.idle": "2025-09-16T16:11:51.580970Z",
     "shell.execute_reply": "2025-09-16T16:11:51.580403Z",
     "shell.execute_reply.started": "2025-09-16T16:11:49.268796Z"
    },
    "trusted": true
   },
   "outputs": [],
   "source": [
    "from scipy.stats import entropy\n",
    "\n",
    "def kl_divergence(p, q, eps=1e-12):\n",
    "    p = np.asarray(p) + eps\n",
    "    q = np.asarray(q) + eps\n",
    "    return np.sum(p * np.log(p / q))\n",
    "\n",
    "def js_divergence(p, q):\n",
    "    p = np.asarray(p)\n",
    "    q = np.asarray(q)\n",
    "    m = 0.5 * (p + q)\n",
    "    return 0.5 * kl_divergence(p, m) + 0.5 * kl_divergence(q, m)\n",
    "\n",
    "def l1_distance(p, q):\n",
    "    return np.sum(np.abs(p - q))\n",
    "\n",
    "def entropy_diff(p, q):\n",
    "    return entropy(q) - entropy(p)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T09:58:27.757665Z",
     "iopub.status.busy": "2025-09-16T09:58:27.757087Z",
     "iopub.status.idle": "2025-09-16T09:58:27.772155Z",
     "shell.execute_reply": "2025-09-16T09:58:27.771487Z",
     "shell.execute_reply.started": "2025-09-16T09:58:27.757639Z"
    },
    "trusted": true
   },
   "outputs": [],
   "source": [
    "def run_fedchill_het(metric=\"ratio\"):\n",
    "    print_flag = False\n",
    "\n",
    "    client_train_loaders, client_val_loaders, test_loader, client_class_distributions = load_and_partition_data(\n",
    "        num_clients=NUM_OF_CLIENTS, \n",
    "        alpha=ALPHA, \n",
    "        batch_size=BATCH_SIZE, \n",
    "        frac=FRAC, \n",
    "        rand_seed=42\n",
    "    )\n",
    "    num_clients = len(client_train_loaders)\n",
    "    num_clients = len(client_train_loaders)\n",
    "\n",
    "    \n",
    "    global_distribution = calculate_global_distribution(client_class_distributions)\n",
    "    \n",
    "    \n",
    "    client_params = [AdaptiveClientHyperparams() for _ in range(num_clients)]\n",
    "    \n",
    "    for i, distribution in enumerate(client_class_distributions):\n",
    "        p = np.array([distribution.get(c, 0) for c in range(10)])\n",
    "        q = np.array([global_distribution.get(c, 1/10) for c in range(10)])\n",
    "\n",
    "        if metric == \"ratio\":\n",
    "            het_score = calculate_heterogeneity_score(distribution, global_distribution)\n",
    "        elif metric == \"kl\":\n",
    "            het_score = kl_divergence(p, q)\n",
    "        elif metric == \"js\":\n",
    "            het_score = js_divergence(p, q)\n",
    "        elif metric == \"l1\":\n",
    "            het_score = l1_distance(p, q)\n",
    "        elif metric == \"entropy_diff\":\n",
    "            het_score = entropy_diff(p, q)\n",
    "        else:\n",
    "            raise ValueError(f\"Unknown heterogeneity metric: {metric}\")\n",
    "\n",
    "        client_params[i].update_temp_from_heterogeneity(het_score)\n",
    "        print(f\"Client {i} - Het Score: {het_score:.4f}, Initial temp: {client_params[i].get_temp():.4f}\")\n",
    "    \n",
    "    \n",
    "    global_model = SimpleCNN().to(device)\n",
    "    round_server_acc_list = []\n",
    "    round_client_acc_list = [[] for _ in range(num_clients)]\n",
    "    \n",
    "    \n",
    "    client_temp_history = [[] for _ in range(num_clients)]\n",
    "    client_lr_history = [[] for _ in range(num_clients)]\n",
    "\n",
    "    client_models = [copy.deepcopy(global_model).to(device) for _ in range(num_clients)]\n",
    "    client_optimizers = [optim.SGD(model.parameters(), lr=client_params[i].get_lr()) for i, model in enumerate(client_models)]\n",
    "\n",
    "    for comm_round in range(COMM_ROUND):\n",
    "        print(f\"\\n{'='*20} COMMUNICATION ROUND {comm_round+1} {'='*20}\")\n",
    "\n",
    "        \n",
    "        selected_clients = list(range(num_clients))\n",
    "\n",
    "        \n",
    "        if comm_round == COMM_ROUND - 1:\n",
    "            print_flag = True\n",
    "            print(\"\\n--- LOCAL TRAINING OF CLIENTS ---\")\n",
    "\n",
    "        for client_idx in selected_clients:\n",
    "            \n",
    "            client_models[client_idx].load_state_dict(global_model.state_dict())\n",
    "            \n",
    "            \n",
    "            curr_lr = client_params[client_idx].get_lr()\n",
    "            curr_temp = client_params[client_idx].get_temp()\n",
    "            \n",
    "            \n",
    "            client_temp_history[client_idx].append(curr_temp)\n",
    "            client_lr_history[client_idx].append(curr_lr)\n",
    "            \n",
    "            \n",
    "            client_optimizers[client_idx] = optim.SGD(client_models[client_idx].parameters(), lr=curr_lr)\n",
    "\n",
    "            if comm_round == COMM_ROUND - 1:\n",
    "                print(f\"\\n--- Client {client_idx} Local Training with T={curr_temp:.4f}, LR={curr_lr:.6f} ---\")\n",
    "\n",
    "            \n",
    "            client_train_with_temp(\n",
    "                client_models[client_idx],\n",
    "                client_train_loaders[client_idx],\n",
    "                client_optimizers[client_idx],\n",
    "                LOCAL_EPOCHS,\n",
    "                temperature=curr_temp,\n",
    "                print_flag=print_flag\n",
    "            )\n",
    "\n",
    "            \n",
    "            local_acc = evaluate_model(client_models[client_idx], client_val_loaders[client_idx])\n",
    "            if comm_round == COMM_ROUND - 1:\n",
    "                print(f\"Client {client_idx} Private Data Validation Accuracy: {local_acc:.2f}%\")\n",
    "\n",
    "            \n",
    "            client_testset_acc = evaluate_model(client_models[client_idx], test_loader)\n",
    "            round_client_acc_list[client_idx].append(client_testset_acc)\n",
    "            \n",
    "            \n",
    "            adjusted, param_type, new_value = client_params[client_idx].record_performance(local_acc)\n",
    "            if adjusted and (comm_round == COMM_ROUND - 1 or print_flag):\n",
    "                print(f\"Client {client_idx} - {param_type} to {new_value:.6f}\")\n",
    "\n",
    "        \n",
    "        aggregate_models(global_model, client_models, selected_clients, client_train_loaders)\n",
    "\n",
    "        \n",
    "        server_test_val_acc = evaluate_model(global_model, test_loader)\n",
    "        if comm_round == COMM_ROUND - 1:\n",
    "            print(f\"\\n--- Server Evaluation ---\")\n",
    "            print(f\"Server Test Data Validation Accuracy: {server_test_val_acc:.2f}%\")\n",
    "\n",
    "        round_server_acc_list.append(server_test_val_acc)\n",
    "\n",
    "\n",
    "    \n",
    "    client_final_accs = []\n",
    "    print(\"\\n--- Clients Final Evaluation on Private Data sets ---\")\n",
    "    for i in range(num_clients):\n",
    "        local_acc = evaluate_model(client_models[i], client_val_loaders[i])\n",
    "        client_final_accs.append(local_acc)\n",
    "        print(f\"Client {i} Final Private Data Validation Accuracy: {local_acc:.2f}%\")\n",
    "\n",
    "    final_test_acc = evaluate_model(global_model, test_loader)\n",
    "    print(f\"\\nFinal Server Test Accuracy: {final_test_acc:.2f}%\")\n",
    "\n",
    "    return client_final_accs, final_test_acc, global_model"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Clients 10, Frac 0.1, Rounds 30, S=-3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T12:34:35.714470Z",
     "iopub.status.busy": "2025-09-16T12:34:35.714083Z",
     "iopub.status.idle": "2025-09-16T12:34:35.718841Z",
     "shell.execute_reply": "2025-09-16T12:34:35.718105Z",
     "shell.execute_reply.started": "2025-09-16T12:34:35.714419Z"
    },
    "trusted": true
   },
   "outputs": [],
   "source": [
    "BATCH_SIZE = 64\n",
    "LEARNING_RATE = 0.01\n",
    "LOCAL_EPOCHS = 5\n",
    "NUM_OF_CLIENTS = 10\n",
    "COMM_ROUND = 30\n",
    "ALPHA = 0.5  \n",
    "FRAC = 0.1   "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T09:58:49.269652Z",
     "iopub.status.busy": "2025-09-16T09:58:49.268939Z",
     "iopub.status.idle": "2025-09-16T11:20:03.185351Z",
     "shell.execute_reply": "2025-09-16T11:20:03.184480Z",
     "shell.execute_reply.started": "2025-09-16T09:58:49.269626Z"
    },
    "trusted": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = ratio\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 0.9007, Initial temp: 0.0671\n",
      "Client 1 - Het Score: 0.7295, Initial temp: 0.1121\n",
      "Client 2 - Het Score: 0.8682, Initial temp: 0.0739\n",
      "Client 3 - Het Score: 0.8552, Initial temp: 0.0769\n",
      "Client 4 - Het Score: 0.5698, Initial temp: 0.1810\n",
      "Client 5 - Het Score: 0.5956, Initial temp: 0.1675\n",
      "Client 6 - Het Score: 0.7451, Initial temp: 0.1070\n",
      "Client 7 - Het Score: 0.6556, Initial temp: 0.1399\n",
      "Client 8 - Het Score: 1.0000, Initial temp: 0.0500\n",
      "Client 9 - Het Score: 1.0000, Initial temp: 0.0500\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.0537, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3174\n",
      "Epoch [2/5], Loss: 0.1923\n",
      "Epoch [3/5], Loss: 0.2270\n",
      "Epoch [4/5], Loss: 0.1785\n",
      "Epoch [5/5], Loss: 0.2066\n",
      "Client 0 Private Data Validation Accuracy: 62.50%\n",
      "\n",
      "--- Client 1 Local Training with T=0.0574, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.7587\n",
      "Epoch [2/5], Loss: 0.8987\n",
      "Epoch [3/5], Loss: 0.3367\n",
      "Epoch [4/5], Loss: 0.1586\n",
      "Epoch [5/5], Loss: 0.1604\n",
      "Client 1 Private Data Validation Accuracy: 65.62%\n",
      "\n",
      "--- Client 2 Local Training with T=0.0739, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.5948\n",
      "Epoch [2/5], Loss: 0.2780\n",
      "Epoch [3/5], Loss: 0.1678\n",
      "Epoch [4/5], Loss: 0.1138\n",
      "Epoch [5/5], Loss: 0.1097\n",
      "Client 2 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Client 3 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.7985\n",
      "Epoch [2/5], Loss: 0.1636\n",
      "Epoch [3/5], Loss: 0.0751\n",
      "Epoch [4/5], Loss: 0.0447\n",
      "Epoch [5/5], Loss: 0.1063\n",
      "Client 3 Private Data Validation Accuracy: 92.71%\n",
      "\n",
      "--- Client 4 Local Training with T=0.0741, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2617\n",
      "Epoch [2/5], Loss: 0.1243\n",
      "Epoch [3/5], Loss: 0.1606\n",
      "Epoch [4/5], Loss: 0.1639\n",
      "Epoch [5/5], Loss: 0.1250\n",
      "Client 4 Private Data Validation Accuracy: 59.38%\n",
      "Client 4 - decrease-temp to 0.059296\n",
      "\n",
      "--- Client 5 Local Training with T=0.1072, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.5864\n",
      "Epoch [2/5], Loss: 0.5981\n",
      "Epoch [3/5], Loss: 0.2334\n",
      "Epoch [4/5], Loss: 0.1435\n",
      "Epoch [5/5], Loss: 0.1877\n",
      "Client 5 Private Data Validation Accuracy: 60.16%\n",
      "\n",
      "--- Client 6 Local Training with T=0.0548, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.4211\n",
      "Epoch [2/5], Loss: 0.2780\n",
      "Epoch [3/5], Loss: 0.1824\n",
      "Epoch [4/5], Loss: 0.0831\n",
      "Epoch [5/5], Loss: 0.0959\n",
      "Client 6 Private Data Validation Accuracy: 67.19%\n",
      "\n",
      "--- Client 7 Local Training with T=0.0895, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.5006\n",
      "Epoch [2/5], Loss: 0.1569\n",
      "Epoch [3/5], Loss: 0.0956\n",
      "Epoch [4/5], Loss: 0.2680\n",
      "Epoch [5/5], Loss: 0.0870\n",
      "Client 7 Private Data Validation Accuracy: 73.44%\n",
      "\n",
      "--- Client 8 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2602\n",
      "Epoch [2/5], Loss: 0.1443\n",
      "Epoch [3/5], Loss: 0.1027\n",
      "Epoch [4/5], Loss: 0.0656\n",
      "Epoch [5/5], Loss: 0.1625\n",
      "Client 8 Private Data Validation Accuracy: 71.48%\n",
      "\n",
      "--- Client 9 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.7332\n",
      "Epoch [2/5], Loss: 0.5061\n",
      "Epoch [3/5], Loss: 0.5288\n",
      "Epoch [4/5], Loss: 0.3039\n",
      "Epoch [5/5], Loss: 0.1369\n",
      "Client 9 Private Data Validation Accuracy: 69.53%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 50.02%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 62.50%\n",
      "Client 1 Final Private Data Validation Accuracy: 68.75%\n",
      "Client 2 Final Private Data Validation Accuracy: 70.31%\n",
      "Client 3 Final Private Data Validation Accuracy: 92.71%\n",
      "Client 4 Final Private Data Validation Accuracy: 59.38%\n",
      "Client 5 Final Private Data Validation Accuracy: 57.81%\n",
      "Client 6 Final Private Data Validation Accuracy: 60.94%\n",
      "Client 7 Final Private Data Validation Accuracy: 70.83%\n",
      "Client 8 Final Private Data Validation Accuracy: 72.27%\n",
      "Client 9 Final Private Data Validation Accuracy: 71.88%\n",
      "\n",
      "Final Server Test Accuracy: 50.02%\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = kl\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 1.0227, Initial temp: 0.0500\n",
      "Client 1 - Het Score: 0.6109, Initial temp: 0.1600\n",
      "Client 2 - Het Score: 1.4072, Initial temp: 0.0500\n",
      "Client 3 - Het Score: 1.6845, Initial temp: 0.0500\n",
      "Client 4 - Het Score: 0.5174, Initial temp: 0.2118\n",
      "Client 5 - Het Score: 0.4435, Initial temp: 0.2643\n",
      "Client 6 - Het Score: 0.7640, Initial temp: 0.1011\n",
      "Client 7 - Het Score: 0.3035, Initial temp: 0.4023\n",
      "Client 8 - Het Score: 1.1237, Initial temp: 0.0500\n",
      "Client 9 - Het Score: 0.7487, Initial temp: 0.1058\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.5308\n",
      "Epoch [2/5], Loss: 0.2114\n",
      "Epoch [3/5], Loss: 0.1768\n",
      "Epoch [4/5], Loss: 0.1661\n",
      "Epoch [5/5], Loss: 0.2530\n",
      "Client 0 Private Data Validation Accuracy: 55.94%\n",
      "\n",
      "--- Client 1 Local Training with T=0.0819, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.2877\n",
      "Epoch [2/5], Loss: 0.4255\n",
      "Epoch [3/5], Loss: 0.2403\n",
      "Epoch [4/5], Loss: 0.1438\n",
      "Epoch [5/5], Loss: 0.0958\n",
      "Client 1 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Client 2 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.3084\n",
      "Epoch [2/5], Loss: 0.3544\n",
      "Epoch [3/5], Loss: 0.2418\n",
      "Epoch [4/5], Loss: 0.1344\n",
      "Epoch [5/5], Loss: 0.1301\n",
      "Client 2 Private Data Validation Accuracy: 71.35%\n",
      "\n",
      "--- Client 3 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.4464\n",
      "Epoch [2/5], Loss: 0.0774\n",
      "Epoch [3/5], Loss: 0.0498\n",
      "Epoch [4/5], Loss: 0.0511\n",
      "Epoch [5/5], Loss: 0.0227\n",
      "Client 3 Private Data Validation Accuracy: 92.19%\n",
      "\n",
      "--- Client 4 Local Training with T=0.1695, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.4069\n",
      "Epoch [2/5], Loss: 0.1113\n",
      "Epoch [3/5], Loss: 0.1283\n",
      "Epoch [4/5], Loss: 0.0500\n",
      "Epoch [5/5], Loss: 0.0751\n",
      "Client 4 Private Data Validation Accuracy: 55.21%\n",
      "\n",
      "--- Client 5 Local Training with T=0.1353, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.7731\n",
      "Epoch [2/5], Loss: 0.3782\n",
      "Epoch [3/5], Loss: 0.3261\n",
      "Epoch [4/5], Loss: 0.1360\n",
      "Epoch [5/5], Loss: 0.1504\n",
      "Client 5 Private Data Validation Accuracy: 58.59%\n",
      "\n",
      "--- Client 6 Local Training with T=0.0517, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.6888\n",
      "Epoch [2/5], Loss: 0.5614\n",
      "Epoch [3/5], Loss: 0.2658\n",
      "Epoch [4/5], Loss: 0.1975\n",
      "Epoch [5/5], Loss: 0.1246\n",
      "Client 6 Private Data Validation Accuracy: 62.50%\n",
      "\n",
      "--- Client 7 Local Training with T=0.4023, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.9017\n",
      "Epoch [2/5], Loss: 0.2246\n",
      "Epoch [3/5], Loss: 0.1449\n",
      "Epoch [4/5], Loss: 0.1145\n",
      "Epoch [5/5], Loss: 0.0954\n",
      "Client 7 Private Data Validation Accuracy: 68.23%\n",
      "\n",
      "--- Client 8 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.4609\n",
      "Epoch [2/5], Loss: 0.1390\n",
      "Epoch [3/5], Loss: 0.1255\n",
      "Epoch [4/5], Loss: 0.1363\n",
      "Epoch [5/5], Loss: 0.0965\n",
      "Client 8 Private Data Validation Accuracy: 78.91%\n",
      "\n",
      "--- Client 9 Local Training with T=0.0542, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.7347\n",
      "Epoch [2/5], Loss: 0.2573\n",
      "Epoch [3/5], Loss: 0.1348\n",
      "Epoch [4/5], Loss: 0.1467\n",
      "Epoch [5/5], Loss: 0.0911\n",
      "Client 9 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 47.73%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 55.94%\n",
      "Client 1 Final Private Data Validation Accuracy: 71.88%\n",
      "Client 2 Final Private Data Validation Accuracy: 71.88%\n",
      "Client 3 Final Private Data Validation Accuracy: 91.67%\n",
      "Client 4 Final Private Data Validation Accuracy: 55.73%\n",
      "Client 5 Final Private Data Validation Accuracy: 54.69%\n",
      "Client 6 Final Private Data Validation Accuracy: 62.50%\n",
      "Client 7 Final Private Data Validation Accuracy: 68.75%\n",
      "Client 8 Final Private Data Validation Accuracy: 78.91%\n",
      "Client 9 Final Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "Final Server Test Accuracy: 47.73%\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = js\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 0.2739, Initial temp: 0.4397\n",
      "Client 1 - Het Score: 0.1681, Initial temp: 0.6039\n",
      "Client 2 - Het Score: 0.3461, Initial temp: 0.3541\n",
      "Client 3 - Het Score: 0.4343, Initial temp: 0.2718\n",
      "Client 4 - Het Score: 0.1533, Initial temp: 0.6314\n",
      "Client 5 - Het Score: 0.1192, Initial temp: 0.6994\n",
      "Client 6 - Het Score: 0.1991, Initial temp: 0.5503\n",
      "Client 7 - Het Score: 0.0819, Initial temp: 0.7822\n",
      "Client 8 - Het Score: 0.2538, Initial temp: 0.4671\n",
      "Client 9 - Het Score: 0.1906, Initial temp: 0.5644\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.0922, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.0461\n",
      "Epoch [2/5], Loss: 0.0400\n",
      "Epoch [3/5], Loss: 0.0245\n",
      "Epoch [4/5], Loss: 0.0541\n",
      "Epoch [5/5], Loss: 0.0099\n",
      "Client 0 Private Data Validation Accuracy: 53.44%\n",
      "\n",
      "--- Client 1 Local Training with T=0.3092, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.8412\n",
      "Epoch [2/5], Loss: 0.2039\n",
      "Epoch [3/5], Loss: 0.0721\n",
      "Epoch [4/5], Loss: 0.0326\n",
      "Epoch [5/5], Loss: 0.0364\n",
      "Client 1 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Client 2 Local Training with T=0.1813, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3159\n",
      "Epoch [2/5], Loss: 0.0482\n",
      "Epoch [3/5], Loss: 0.0559\n",
      "Epoch [4/5], Loss: 0.0476\n",
      "Epoch [5/5], Loss: 0.0321\n",
      "Client 2 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Client 3 Local Training with T=0.1739, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2314\n",
      "Epoch [2/5], Loss: 0.0123\n",
      "Epoch [3/5], Loss: 0.0077\n",
      "Epoch [4/5], Loss: 0.0047\n",
      "Epoch [5/5], Loss: 0.0010\n",
      "Client 3 Private Data Validation Accuracy: 90.62%\n",
      "Client 3 - decrease-temp to 0.139145\n",
      "\n",
      "--- Client 4 Local Training with T=0.3233, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1464\n",
      "Epoch [2/5], Loss: 0.0429\n",
      "Epoch [3/5], Loss: 0.0263\n",
      "Epoch [4/5], Loss: 0.0190\n",
      "Epoch [5/5], Loss: 0.0117\n",
      "Client 4 Private Data Validation Accuracy: 63.02%\n",
      "\n",
      "--- Client 5 Local Training with T=0.1833, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.5294\n",
      "Epoch [2/5], Loss: 0.3331\n",
      "Epoch [3/5], Loss: 0.2227\n",
      "Epoch [4/5], Loss: 0.1315\n",
      "Epoch [5/5], Loss: 0.1207\n",
      "Client 5 Private Data Validation Accuracy: 50.00%\n",
      "\n",
      "--- Client 6 Local Training with T=0.3522, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.9256\n",
      "Epoch [2/5], Loss: 0.1227\n",
      "Epoch [3/5], Loss: 0.0549\n",
      "Epoch [4/5], Loss: 0.0441\n",
      "Epoch [5/5], Loss: 0.0316\n",
      "Client 6 Private Data Validation Accuracy: 67.19%\n",
      "\n",
      "--- Client 7 Local Training with T=0.3204, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1893\n",
      "Epoch [2/5], Loss: 0.0468\n",
      "Epoch [3/5], Loss: 0.0267\n",
      "Epoch [4/5], Loss: 0.0218\n",
      "Epoch [5/5], Loss: 0.0119\n",
      "Client 7 Private Data Validation Accuracy: 75.00%\n",
      "\n",
      "--- Client 8 Local Training with T=0.2391, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1241\n",
      "Epoch [2/5], Loss: 0.0098\n",
      "Epoch [3/5], Loss: 0.0089\n",
      "Epoch [4/5], Loss: 0.0048\n",
      "Epoch [5/5], Loss: 0.0081\n",
      "Client 8 Private Data Validation Accuracy: 80.08%\n",
      "\n",
      "--- Client 9 Local Training with T=0.1480, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1968\n",
      "Epoch [2/5], Loss: 0.1484\n",
      "Epoch [3/5], Loss: 0.2561\n",
      "Epoch [4/5], Loss: 0.1575\n",
      "Epoch [5/5], Loss: 0.1906\n",
      "Client 9 Private Data Validation Accuracy: 69.53%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 55.61%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 53.44%\n",
      "Client 1 Final Private Data Validation Accuracy: 67.19%\n",
      "Client 2 Final Private Data Validation Accuracy: 68.75%\n",
      "Client 3 Final Private Data Validation Accuracy: 90.62%\n",
      "Client 4 Final Private Data Validation Accuracy: 64.58%\n",
      "Client 5 Final Private Data Validation Accuracy: 46.09%\n",
      "Client 6 Final Private Data Validation Accuracy: 68.75%\n",
      "Client 7 Final Private Data Validation Accuracy: 75.52%\n",
      "Client 8 Final Private Data Validation Accuracy: 80.08%\n",
      "Client 9 Final Private Data Validation Accuracy: 71.09%\n",
      "\n",
      "Final Server Test Accuracy: 55.61%\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = l1\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 1.2069, Initial temp: 0.0500\n",
      "Client 1 - Het Score: 0.9669, Initial temp: 0.0550\n",
      "Client 2 - Het Score: 1.4128, Initial temp: 0.0500\n",
      "Client 3 - Het Score: 1.6220, Initial temp: 0.0500\n",
      "Client 4 - Het Score: 0.8076, Initial temp: 0.0887\n",
      "Client 5 - Het Score: 0.7334, Initial temp: 0.1108\n",
      "Client 6 - Het Score: 0.9941, Initial temp: 0.0507\n",
      "Client 7 - Het Score: 0.6560, Initial temp: 0.1397\n",
      "Client 8 - Het Score: 1.1727, Initial temp: 0.0500\n",
      "Client 9 - Het Score: 1.0408, Initial temp: 0.0500\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3388\n",
      "Epoch [2/5], Loss: 0.1633\n",
      "Epoch [3/5], Loss: 0.1780\n",
      "Epoch [4/5], Loss: 0.1841\n",
      "Epoch [5/5], Loss: 0.1954\n",
      "Client 0 Private Data Validation Accuracy: 52.81%\n",
      "\n",
      "--- Client 1 Local Training with T=0.0550, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.7451\n",
      "Epoch [2/5], Loss: 0.7028\n",
      "Epoch [3/5], Loss: 0.3194\n",
      "Epoch [4/5], Loss: 0.1924\n",
      "Epoch [5/5], Loss: 0.1847\n",
      "Client 1 Private Data Validation Accuracy: 67.19%\n",
      "\n",
      "--- Client 2 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.8749\n",
      "Epoch [2/5], Loss: 0.3059\n",
      "Epoch [3/5], Loss: 0.2405\n",
      "Epoch [4/5], Loss: 0.1337\n",
      "Epoch [5/5], Loss: 0.1311\n",
      "Client 2 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Client 3 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3913\n",
      "Epoch [2/5], Loss: 0.0963\n",
      "Epoch [3/5], Loss: 0.0611\n",
      "Epoch [4/5], Loss: 0.0505\n",
      "Epoch [5/5], Loss: 0.0332\n",
      "Client 3 Private Data Validation Accuracy: 93.23%\n",
      "\n",
      "--- Client 4 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3563\n",
      "Epoch [2/5], Loss: 0.1685\n",
      "Epoch [3/5], Loss: 0.1058\n",
      "Epoch [4/5], Loss: 0.2343\n",
      "Epoch [5/5], Loss: 0.0844\n",
      "Client 4 Private Data Validation Accuracy: 55.73%\n",
      "\n",
      "--- Client 5 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.2010\n",
      "Epoch [2/5], Loss: 0.7798\n",
      "Epoch [3/5], Loss: 0.3398\n",
      "Epoch [4/5], Loss: 0.2604\n",
      "Epoch [5/5], Loss: 0.2742\n",
      "Client 5 Private Data Validation Accuracy: 54.69%\n",
      "\n",
      "--- Client 6 Local Training with T=0.0507, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.3579\n",
      "Epoch [2/5], Loss: 0.3692\n",
      "Epoch [3/5], Loss: 0.2512\n",
      "Epoch [4/5], Loss: 0.1754\n",
      "Epoch [5/5], Loss: 0.1432\n",
      "Client 6 Private Data Validation Accuracy: 73.44%\n",
      "\n",
      "--- Client 7 Local Training with T=0.0715, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3425\n",
      "Epoch [2/5], Loss: 0.2046\n",
      "Epoch [3/5], Loss: 0.1663\n",
      "Epoch [4/5], Loss: 0.1443\n",
      "Epoch [5/5], Loss: 0.2795\n",
      "Client 7 Private Data Validation Accuracy: 58.33%\n",
      "\n",
      "--- Client 8 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3130\n",
      "Epoch [2/5], Loss: 0.1690\n",
      "Epoch [3/5], Loss: 0.1606\n",
      "Epoch [4/5], Loss: 0.0701\n",
      "Epoch [5/5], Loss: 0.0545\n",
      "Client 8 Private Data Validation Accuracy: 80.86%\n",
      "\n",
      "--- Client 9 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.3439\n",
      "Epoch [2/5], Loss: 0.4928\n",
      "Epoch [3/5], Loss: 0.4074\n",
      "Epoch [4/5], Loss: 0.1691\n",
      "Epoch [5/5], Loss: 0.1369\n",
      "Client 9 Private Data Validation Accuracy: 72.66%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 51.75%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 52.81%\n",
      "Client 1 Final Private Data Validation Accuracy: 71.88%\n",
      "Client 2 Final Private Data Validation Accuracy: 68.75%\n",
      "Client 3 Final Private Data Validation Accuracy: 92.71%\n",
      "Client 4 Final Private Data Validation Accuracy: 55.21%\n",
      "Client 5 Final Private Data Validation Accuracy: 53.12%\n",
      "Client 6 Final Private Data Validation Accuracy: 65.62%\n",
      "Client 7 Final Private Data Validation Accuracy: 59.90%\n",
      "Client 8 Final Private Data Validation Accuracy: 80.47%\n",
      "Client 9 Final Private Data Validation Accuracy: 71.09%\n",
      "\n",
      "Final Server Test Accuracy: 51.75%\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = entropy_diff\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 0.8527, Initial temp: 0.0774\n",
      "Client 1 - Het Score: 0.7037, Initial temp: 0.1211\n",
      "Client 2 - Het Score: 1.3864, Initial temp: 0.0500\n",
      "Client 3 - Het Score: 1.6167, Initial temp: 0.0500\n",
      "Client 4 - Het Score: 0.5006, Initial temp: 0.2227\n",
      "Client 5 - Het Score: 0.4251, Initial temp: 0.2793\n",
      "Client 6 - Het Score: 0.9316, Initial temp: 0.0611\n",
      "Client 7 - Het Score: 0.3201, Initial temp: 0.3827\n",
      "Client 8 - Het Score: 1.0807, Initial temp: 0.0500\n",
      "Client 9 - Het Score: 0.8083, Initial temp: 0.0885\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2593\n",
      "Epoch [2/5], Loss: 0.1429\n",
      "Epoch [3/5], Loss: 0.1565\n",
      "Epoch [4/5], Loss: 0.0988\n",
      "Epoch [5/5], Loss: 0.1693\n",
      "Client 0 Private Data Validation Accuracy: 47.50%\n",
      "\n",
      "--- Client 1 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 2.9011\n",
      "Epoch [2/5], Loss: 0.9425\n",
      "Epoch [3/5], Loss: 0.4188\n",
      "Epoch [4/5], Loss: 0.2357\n",
      "Epoch [5/5], Loss: 0.2488\n",
      "Client 1 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Client 2 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 2.2974\n",
      "Epoch [2/5], Loss: 0.4495\n",
      "Epoch [3/5], Loss: 0.2884\n",
      "Epoch [4/5], Loss: 0.2494\n",
      "Epoch [5/5], Loss: 0.2723\n",
      "Client 2 Private Data Validation Accuracy: 66.15%\n",
      "\n",
      "--- Client 3 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.7073\n",
      "Epoch [2/5], Loss: 0.0678\n",
      "Epoch [3/5], Loss: 0.0650\n",
      "Epoch [4/5], Loss: 0.0623\n",
      "Epoch [5/5], Loss: 0.0455\n",
      "Client 3 Private Data Validation Accuracy: 93.23%\n",
      "\n",
      "--- Client 4 Local Training with T=0.1140, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1927\n",
      "Epoch [2/5], Loss: 0.1091\n",
      "Epoch [3/5], Loss: 0.0701\n",
      "Epoch [4/5], Loss: 0.0666\n",
      "Epoch [5/5], Loss: 0.0355\n",
      "Client 4 Private Data Validation Accuracy: 61.46%\n",
      "\n",
      "--- Client 5 Local Training with T=0.1144, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.6618\n",
      "Epoch [2/5], Loss: 0.4542\n",
      "Epoch [3/5], Loss: 0.2255\n",
      "Epoch [4/5], Loss: 0.1722\n",
      "Epoch [5/5], Loss: 0.1105\n",
      "Client 5 Private Data Validation Accuracy: 62.50%\n",
      "\n",
      "--- Client 6 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.4078\n",
      "Epoch [2/5], Loss: 0.3633\n",
      "Epoch [3/5], Loss: 0.2024\n",
      "Epoch [4/5], Loss: 0.1407\n",
      "Epoch [5/5], Loss: 0.1727\n",
      "Client 6 Private Data Validation Accuracy: 71.88%\n",
      "\n",
      "--- Client 7 Local Training with T=0.1254, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.5827\n",
      "Epoch [2/5], Loss: 0.1764\n",
      "Epoch [3/5], Loss: 0.0902\n",
      "Epoch [4/5], Loss: 0.1149\n",
      "Epoch [5/5], Loss: 0.1072\n",
      "Client 7 Private Data Validation Accuracy: 64.06%\n",
      "\n",
      "--- Client 8 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3771\n",
      "Epoch [2/5], Loss: 0.1313\n",
      "Epoch [3/5], Loss: 0.0972\n",
      "Epoch [4/5], Loss: 0.0556\n",
      "Epoch [5/5], Loss: 0.1481\n",
      "Client 8 Private Data Validation Accuracy: 75.78%\n",
      "\n",
      "--- Client 9 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 3.5599\n",
      "Epoch [2/5], Loss: 0.6940\n",
      "Epoch [3/5], Loss: 0.4159\n",
      "Epoch [4/5], Loss: 0.3814\n",
      "Epoch [5/5], Loss: 0.2176\n",
      "Client 9 Private Data Validation Accuracy: 69.53%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 48.64%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 47.50%\n",
      "Client 1 Final Private Data Validation Accuracy: 68.75%\n",
      "Client 2 Final Private Data Validation Accuracy: 66.15%\n",
      "Client 3 Final Private Data Validation Accuracy: 92.19%\n",
      "Client 4 Final Private Data Validation Accuracy: 63.54%\n",
      "Client 5 Final Private Data Validation Accuracy: 60.16%\n",
      "Client 6 Final Private Data Validation Accuracy: 68.75%\n",
      "Client 7 Final Private Data Validation Accuracy: 63.02%\n",
      "Client 8 Final Private Data Validation Accuracy: 75.78%\n",
      "Client 9 Final Private Data Validation Accuracy: 71.09%\n",
      "\n",
      "Final Server Test Accuracy: 48.64%\n"
     ]
    },
    {
     "data": {
      "image/png": "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jiQzZMiQbKc7nU7Trl07Y1mWueOOO8xrr71munbtaiSZYcOGecz7+OOPG0nmqquuMs8995x5+eWXzc0332wefvhh1zzdunUzvXv3Ns8995yZOnWquemmm4wk89BDD3mNPzY29oL5G2PMqFGjjGVZHp+T3HA4HCY0NNQ0adLEHD58+ILz5vTzl90+YfDgwSYyMtKkpKR4zD9r1iwjyaxZsybfXut8vv/+eyPJfP3118YYYwYPHmzq1KnjNd+SJUtMUFCQiY2NNWPHjjVTp041999/v+nQoYNrno0bN5rw8HBTtmxZM2rUKPP666+bkSNHmnr16rnmyW6dZuwvcjqenO6TL1SPed3uofihSUKRkZycbCR5bcyzs2HDBiPJ3HHHHR7xhx56yEgyX375pSsWGxtrJJnvvvvOFfviiy+MJBMaGmr++OMPV/z111/3+vKT8QXkvvvuc8WcTqfp0qWLCQoKMgcPHnTFT58+7ZFPamqqufzyy027du084pKMzWYzmzZt8hpb1iYpIiLivF9MUlNTTfny5c3ll19uzpw544r/73//M5LMmDFjvMaSdYPeqFEj06RJk2xfwxhjDhw4YIKCgsy1117r0US+9tprRpJ5++23XbGMHZn7e5OdjHnP98+9Sfr666+NJPPuu+96LOfzzz/3itetW9e0bt3a6zWffPJJExYWZrZt2+YR/89//mPsdrvZtWuXMSazSQoPDzcHDhzwmLdhw4amfPnyHl+UNm7caGw2mxkwYIAr1rVrV1OiRAnz119/uWLbt283AQEBHjv7nTt3Grvdbp5++mmP1/nll19MQECAR7x169ZGkpk9e7YrlpKSYipWrGh69uzpir3zzjvGZrO5vthkmDZtmpFkvv32W1fMvUkyxph58+Z5fRaMOfdlLDo62vTp08cj/uKLLxrLssyOHTvM+WQ0ScYY06tXL9O+fXvXcitWrGjGjRvns0nK6TozJmefw5deeumCNZrbJkmSmTZtmse8OV0HOcnHl4vZFq5atcoVO3DggAkODjYPPvjgBV/rQk3SwoULjSTz1FNPecR79eplLMsyv/32mzHmXP3bbDbTvXt3j22JMee2rRmyrkdjjLn77rtNiRIlzNmzZ12xnDZJaWlp5pZbbjFBQUEmLCzMY5+QG2PGjDGSTFhYmLn++uvN008/bdatW+c1X24+f9ntEzL2VVm/nHfu3NnjR8X8eK3zGTp0qImJiXGtnyVLlhhJHs1Venq6iYuLM7GxsR4/6hjjuV5btWplSpUq5bHvzTpPbpuk7MaTk21BTuoxr9s9FD+cboci4/jx45KkUqVK5Wj+zz77TJI0YsQIj/iDDz4oSV7XLtWpU0ctWrRwPb7iiiskSe3atVPVqlW94jt27PB6Tfe7OmUc3k9NTdWyZctc8dDQUNf/jx49quTkZF1zzTVep8ZJUuvWrVWnTp0LjPTcdT0//PCD9uzZ43P62rVrdeDAAd17770e52B36dJFCQkJPq/juueeezweX3PNNT7H7G7ZsmVKTU3VsGHDPC7wv/POOxUeHp7n68U++ugjLV261OtfhQoVPOabN2+eIiIi1LFjRx06dMj1r0mTJipZsqS++uqrC77WvHnzdM0116h06dIey+jQoYMcDodWrVrlMX/Pnj0VFRXlerx3715t2LBBAwcOVJkyZVzx+vXrq2PHjq76dDgcWrZsmbp166bKlSu75qtRo4auv/56j9eYP3++nE6nevfu7ZFTxYoVVbNmTa9xlSxZUrfccovrcVBQkJo3b+6xHufNm6fatWsrISHBY5nt2rWTpBy9V1nZbDb1799fn3zyiU6cOOGKv/vuu7rqqqsUFxeX42XdfPPNWrFihfbt26cvv/xS+/bty/ZUu9yss5x8DjOul/v444+9Tte7WMHBwV6nRuV0HVxsPhezLbzmmmtcj6OiolSrVq0Lfv5zmovdbtf999/vlYsxRosXL5YkLVy4UE6nU2PGjPHYlkjyOJXKfT2eOHFChw4d0jXXXKPTp09ry5Ytuc5v5MiRWrx4sX755RddccUV6ty5szZs2OCavnfvXlmW5XG9ki/jxo3T3Llz1ahRI33xxRd65JFH1KRJEzVu3NjjdNvcfv587RPatWuncuXK6YMPPnDFjh49qqVLl6pPnz75+lrZSU9P1wcffKA+ffq41k+7du1Uvnx5vfvuu6751q9fr6SkJA0bNszretSM5x08eFCrVq3S4MGDPfa97vNcjOzGk5NtQU7qMT+3eygeuHEDiozw8HBJ8tj4nM8ff/whm83mdee0ihUrKjIyUn/88YdHPOvGOCIiQpIUExPjM+5+bYt0bgNZvXp1j9hll10mSR7XK/zvf//TU089pQ0bNnicB+5r45/Tjeqzzz6r2267TTExMWrSpIk6d+6sAQMGuPLJGKuvc6ITEhL0zTffeMRCQkI8vvBLUunSpb3GnFV2rxMUFKTq1at7vee51apVK5UrV84rnvXi2+3btys5Odnr3P8MOblwevv27fr555+93ofslpF1XZ3vPa9du7a++OILnTp1SsePH9eZM2d83uEva2z79u0yxqhmzZo+cwoMDPR4HB0d7VVXpUuX1s8//+yxzF9//TXH48ypAQMGaOLEiVqwYIEGDBigrVu3at26dZo2bVqultO5c2eVKlVKH3zwgTZs2KBmzZqpRo0aPm87npt1lpPPYZ8+ffTWW2/pjjvu0H/+8x+1b99ePXr0UK9evby+KOVUlSpVvG7okdN1cLH55HVbKOXs858Tf/zxhypXruz1Y1ft2rVd06Vzf+bBZrNd8Ev6pk2b9Oijj+rLL790/ZCWITk5OVe5/fXXX3rllVc0fvx4XXbZZVq4cKFat26ta6+9Vl9//bVq1aqlxMRESZk/lp1Pv3791K9fPx0/flw//PCDZs6cqblz56pr165KTExUSEhIrj9/vvYJAQEB6tmzp+bOnauUlBQFBwdr/vz5SktL82iS8uO1srNkyRIdPHhQzZs312+//eaKt23bVu+9954mTpwom82m33//XZJ0+eWXZ7usjGb8fPNcjOzGk5NtQU7rMb+2eygeaJJQZISHh6ty5cqunVRO5fSXJ7vdnqu48XGR8YV8/fXX+te//qVWrVppypQpqlSpkgIDAzVjxgyfF7i6/8J1Pr1799Y111yjBQsWaMmSJXruuec0ceJEzZ8/3+toRE5kN+biwul0ev2C6S67LwlZl9GxY0eNHDnS5/SMBjhDTtdVXjidTlmWpcWLF/tcRyVLlvR4nJPadTqdqlevnl588UWf82b9kSCn6tSpoyZNmmjOnDkaMGCA5syZo6CgIK871V1IcHCwevTooVmzZmnHjh3nvcFHTtdZTj+HoaGhWrVqlb766istWrRIn3/+uT744AO1a9dOS5Yskd1uz3b7kt2NA3zVSU7XQU7yOZ+8bgsvZptXkI4dO6bWrVsrPDxcTzzxhOLj4xUSEqKffvpJDz/8cK6P/v3www9yOBy68sorJZ07a2Hx4sVq2bKlOnTooK+//lpvvPGGGjRokKsv8OHh4erYsaM6duyowMBAzZo1Sz/88INat26d689fdtuZvn376vXXX9fixYvVrVs3/fe//1VCQoIaNGjgmie/XsuXjG1tdp/vlStXqm3btjleXk7kx2cvt/vkC8mv7R6KB5okFCk33HCD3njjDa1evdrj1DhfYmNj5XQ6tX37dtevlJK0f/9+HTt2TLGxsfmam9Pp1I4dOzy+PG/btk2SXHfg+eijjxQSEqIvvvjC425JM2bMyPPrV6pUSffee6/uvfdeHThwQI0bN9bTTz+t66+/3jXWrVu3uk6tyLB169Z8ey/cX8f9qFpqaqqSkpLUoUOHfHmdC4mPj9eyZcvUsmXLC+7os9vRxsfH6+TJkxeds/t7kdWWLVtUrlw5hYWFuW7p7v7ra4assfj4eBljFBcX59WkXaz4+Hht3LhR7du3z/WpLBeaf8CAARoxYoT27t3rum136dKlc53jzTffrLfffls2m019+/bNdr6crrPcfA5tNpvat2+v9u3b68UXX9QzzzyjRx55RF999ZU6dOjgGo/7Hfgk5eqoaW7WwYXy8aWwt4XnExsbq2XLlunEiRMeR5MyTo3LyCU+Pl5Op1ObN29Ww4YNfS5rxYoVOnz4sObPn69WrVq54klJSReVW8Z7736nyAoVKuiLL75Qy5Yt1bp1a+3evVvz58+/qOVLUtOmTTVr1izt3btXUt4+f+5atWqlSpUq6YMPPtDVV1+tL7/80uNOcPn5WlmdOnVKH3/8sfr06aNevXp5Tb///vv17rvvqm3btoqPj5ckJSYmZluvGfuOC/0gWrp0aa/PnZS7z15OtwU5qccM+bXdQ9HHNUkoUkaOHKmwsDDdcccd2r9/v9f033//3XUb0c6dO0uSJk2a5DFPxq9oXbp0yff8XnvtNdf/jTF67bXXFBgYqPbt20uS65dn91+6du7cqYULF170azocDq/TSsqXL6/KlSu7Th1o2rSpypcvr2nTpnmcTrB48WL9+uuv+fZedOjQQUFBQXrllVc8fnWePn26kpOTC+Q996V3795yOBx68sknvaalp6d77FjDwsJ87mh79+6t1atX64svvvCaduzYMaWnp583h0qVKqlhw4aaNWuWx/ITExO1ZMkSV33a7XZ16NBBCxcu9Lim7LfffnNdn5GhR48estvtGjdunNev+sYYHT58+Lw5+dK7d2/99ddfevPNN72mnTlzRqdOncr2uWFhYZK8G4QM/fr1k2VZeuCBB7Rjxw6P66Nyo23btnryySf12muvqWLFitnOl9N1ltPP4ZEjR7yWk/EFKeNzlPGlz/16J4fDkas/aJ3TdZCTfHzxx7bwfLk4HA6PbaUkvfTSS7Isy3Xku1u3brLZbHriiSe8jghl1H7GES/3z0JqaqqmTJlyUbldffXVCg4O1oQJE3T69GlXPD4+XpMmTdKuXbsUERGh1q1bn3c5p0+f1urVq31Oy/hMZ5yGm5fPnzubzaZevXrp008/1TvvvKP09HSPU+3y87WyWrBggU6dOqUhQ4aoV69eXv9uuOEGffTRR0pJSVHjxo0VFxenSZMmeW03MtZjVFSUWrVqpbffflu7du3yOY90br0kJyd7nD68d+9eLViwIMe553RbkJN6zJBf2z0UfRxJQpESHx+vuXPnqk+fPqpdu7YGDBigyy+/XKmpqfruu+80b948199xadCggW677Ta98cYbrtMyfvzxR82aNUvdunXL90P/ISEh+vzzz3Xbbbfpiiuu0OLFi7Vo0SKNHj3adXpXly5d9OKLL+q6667TzTffrAMHDmjy5MmqUaOGx4Y+N06cOKHo6Gj16tVLDRo0UMmSJbVs2TKtWbNGL7zwgqRz16pMnDhRgwYNUuvWrdWvXz/t379fL7/8sqpVq6bhw4fny3sQFRWlUaNGady4cbruuuv0r3/9S1u3btWUKVPUrFmzQttZtG7dWnfffbfGjx+vDRs26Nprr1VgYKC2b9+uefPm6eWXX3b94tmkSRNNnTpVTz31lGrUqKHy5curXbt2+r//+z998sknuuGGGzRw4EA1adJEp06d0i+//KIPP/xQO3fu9Hl9lLvnnntO119/vVq0aKHbb79dZ86c0auvvqqIiAiP08Yef/xxLVmyRC1bttS///1v15fIyy+/3OOi8fj4eD311FMaNWqUdu7cqW7duqlUqVJKSkrSggULdNddd3n8TZScuPXWW/Xf//5X99xzj7766iu1bNlSDodDW7Zs0X//+1998cUXatq0qc/nNmzYUHa7XRMnTlRycrKCg4NdF2tL5+rhuuuu07x58xQZGXnRX8ZtNpseffTRC86X03WW08/hE088oVWrVqlLly6KjY3VgQMHNGXKFEVHR+vqq6+WJNWtW1dXXnmlRo0apSNHjqhMmTJ6//33L9hEu8vpOshJPr4U9rZw7dq1euqpp7zibdq0UdeuXdW2bVs98sgj2rlzpxo0aKAlS5bo448/1rBhw1xNZ40aNfTII4/oySef1DXXXKMePXooODhYa9asUeXKlTV+/HhdddVVKl26tG677Tbdf//9sixL77zzzkWfFhgVFaXx48drxIgRqlevngYPHqyKFStq7dq1mjVrlq688kr99NNP6tWrlxYvXux1DWCG06dP66qrrtKVV16p6667TjExMTp27JgWLlyor7/+Wt26dVOjRo0k5e3zl1WfPn306quvauzYsapXr57HUcP8fi137777rsqWLaurrrrK5/R//etfevPNN7Vo0SL16NFDU6dOVdeuXdWwYUMNGjRIlSpV0pYtW7Rp0ybXDxyvvPKKrr76ajVu3Fh33XWX4uLitHPnTi1atMi1Tezbt68efvhhde/eXffff79Onz6tqVOn6rLLLvN5IyRfcrotyEk9Zsiv7R6KgcK/oR5wYdu2bTN33nmnqVatmgkKCjKlSpUyLVu2NK+++qrHbV/T0tLMuHHjTFxcnAkMDDQxMTFm1KhRHvMYc+62t1n/zo4xvm9n6+vWwxm3LP79999df4eiQoUKZuzYsV63C50+fbqpWbOmCQ4ONgkJCWbGjBnZ3rI0u1vpyu0W4CkpKeb//u//TIMGDUypUqVMWFiYadCggc+/afTBBx+YRo0ameDgYFOmTBnTv39/s3v3bo953G+/7M5Xjtl57bXXTEJCggkMDDQVKlQw//73v71u93oxtwDPbt7s1t8bb7xhmjRpYkJDQ02pUqVMvXr1zMiRI82ePXtc8+zbt8906dLFlCpVykjyuB34iRMnzKhRo0yNGjVMUFCQKVeunLnqqqvM888/7/pbS77qwd2yZctMy5YtTWhoqAkPDzddu3Y1mzdv9ppv+fLlplGjRiYoKMjEx8ebt956yzz44IMmJCTEa96PPvrIXH311SYsLMyEhYWZhIQEM2TIELN161bXPK1btzZ169b1eq6v2+ampqaaiRMnmrp165rg4GBTunRp06RJEzNu3DiTnJzs8T673wLcGGPefPNNU716dWO3233eDvy///2vkWTuuusun++PL9nVoLvs3vecrDNjcvY5XL58ubnxxhtN5cqVTVBQkKlcubLp16+f1y3Gf//9d9OhQwcTHBzs+vtCGX8zJestwH2tE2Nytg5ymo8ved0Wtm7d2uet8rPSeW7T/+STTxpj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89uFTJ8+XZs2bfK4zsbpdOrtt9/W7bffft4cLpSjdO6Uuazf09PS0rzmy7g5QoadO3fqhhtu0L///W89/fTTKlOmjL755hvdfvvtSk1NVYkSJS742o0aNVKDBg00e/ZsXXvttdq0aZMWLSq8o6R+b5IAAADwz3TdddcpNTVVlmWpU6dOXtPr1Kmj4OBg7dq1S61bt/a5jG+//VZXXXWV7r33Xlfs999/L7CcJal27dr6888/9eeff3qcbnfs2DHXUaD69etr+fLlPq+z2rJliw4fPqwJEya4np/1lu8X8ssvv2jt2rVasWKFx5GgI0eOqE2bNtqyZYtq1qyp0NBQLV++3OfpdvXr19dbb72lI0eO+DyaFBUVpcTERI/Yhg0bvBrOrNatWyen06kXXnjBdeOSrNeIZbw/vv50T4Y77rhDkyZN0l9//aUOHTp4HLkraDRJhawwzxP2t4s5TxkAAPxz2O12/frrr67/Z1WqVCk99NBDGj58uJxOp66++molJyfr22+/VXh4uG677TbVrFlTs2fP1hdffKG4uDi98847WrNmjeLi4vKcn8Ph0IYNGzxiwcHB6tChg+rVq6f+/ftr0qRJSk9P17333qvWrVuradOmkqSxY8eqffv2io+PV9++fZWenq7PPvtMDz/8sKpWraqgoCC9+uqruueee5SYmKgnn3wyV7lNnz5dzZs3V6tWrbymNWvWTNOnT9dzzz2nhx9+WCNHjlRQUJBatmypgwcPatOmTbr99tvVr18/PfPMM+rWrZvGjx+vSpUqaf369apcubJatGihdu3a6bnnntPs2bPVokULzZkzR4mJiR6nOvpSo0YNpaWl6dVXX1XXrl09buiQYdSoUapXr57uvfde3XPPPQoKCtJXX32lm266yXXa5c0336yHHnpIb775pmbPnp2r9yevaJIA4B/un/LjDT/cAEXThW6s9eSTTyoqKkrjx4/Xjh07FBkZqcaNG2v06NGSpLvvvlvr169Xnz59ZFmW+vXrp3vvvVeLFy/Oc24nT570agji4+P122+/6eOPP9Z9992nVq1ayWaz6brrrtOrr77qmq9NmzaaN2+ennzySU2YMEHh4eGuhiYqKkozZ87U6NGj9corr6hx48Z6/vnn9a9//StHeaWmpmrOnDl6+OGHfU7v2bOnXnjhBT3zzDN67LHHFBAQoDFjxmjP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1ZHDB3XtDd0kSRs3btTJkydVtmxZj+edOXNGv//+u+txbGzsBRskSXr77bd1yy23uB7fcsstat26tV599VWVKlVKGzZsUKNGjVwNUlYbNmzQnXfeecHXuZCmTZt6PHY4HHrmmWf03//+V3/99ZdSU1OVkpKiEiXOvW+//vqrUlJS1L59e5/LCwkJcZ0+2Lt3b/30009KTEzUJ598kudcs0OTBAAAgEIVFhbmcQTjfAICAj0eW5Yl8/fpYRvX/ajR99+lf4/4j65q3V4lw8P1+cfz9c6br+U6p5hqcYqpFqce/Qbojvse1I2tm+qLTxaoW5/+On3qlP794H/U/rquXs8LDs68Rig01LvZ69Ktlxb/3SQtXvihWrZur8jS55qUkydPqlKlSlqxYoXX8yIjI13/P18TmWHz5s36/vvv9eOPP+rhhzOPbjkcDr3//vu68847FRoaet5lXGi6ZVkyxnjEfF3/lTXf5557Ti+//LImTZqkevXqKSwsTMOGDXPdhOJCryudO+WuYcOG2r17t2bMmKF27dopNjb2gs+7WFyTBAAAgGJpw9ofValKjO68/yHVbdBIsXHx2vvXn3lebpWYqgoJDdWZM+duR167Xn3t/P03VY2r7vXPZjv/1+nru92k37b+qs0/b9DSzz5W5+43uaY1btxY+/btU0BAgGrUqOHxr1y5crnKefr06WrVqpU2btyoDRs2uP6NGDFC06efu3lE/fr1tWHDBo/rndzVr1//vDdCiIqK0t69e12Pt2/frtOnT18wt2+//VY33nijbrnlFjVo0EDVq1fXtm3bXNNr1qyp0NDQ8752vXr11LRpU7355puaO3euBg8efMHXzQuOJAEAAKBYio2rrn17dmvxxx/p8gaNterLJfry8//lahlTX5ygs2dO6+p216pSlRidOJ6suW+/rvS0dLW4pq0k6a4HRur+QX1VqUq0OnT+l2w2m7ZuTtTvW3/V0JGPnnf5VWKqqkHT5nr8/+6T0+FUm47Xu6Z16NBBLVq0ULdu3fTss8/qsssu0549e1w3UMh62lp20tLS9M477+iJJ57Q5Zdf7jHtjjvu0IsvvqhNmzapX79+euaZZ9StWzeNHz9elSpV0vr161W5cmW1aNFCY8eOVfv27RUfH6++ffsqPT1dn332mevIVLt27fTaa6+pRYsWcjgcevjhhxUYGOgr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IiJ0++23a8SIESpTpozCw8N13333qUWLFtzZDgAAAECB8WuTtHv3bvXr10+HDx9WVFSUrr76an3//feKioqSJL300kuy2Wzq2bOnUlJS1KlTJ02ZMsWfKQMAAAC4xPm1SXr//ffPOz0kJESTJ0/W5MmTCykjAAAAAP90ReqaJAAAAADwN5okAAAAAHBTpG4BDgAAAPxjPB7h7wwKz+PJF56nCOFIEgAAAAC4oUkCAAAAADc0SQAAAADghiYJAAAAANzQJAEAAACAG+5uB6Do+6fc/aeY3fkHAIBLFUeSAAAAAMANTRIAAAAAuKFJAgAAAAA3NEkAAAAA4IYmCQAAAADc0CQBAAAAgBuaJAAAAABwQ5MEAAAAAG5okgAAAADADU0SAAAAALihSQIAAAAANzRJAAAAAOCGJgkAAAAA3NAkAQAAAIAbmiQAAAAAcEOTBAAAAABuaJIAAAAAwA1NEgAAAAC4oUkCAAAAADc0SQAAAADghiYJAAAAANzQJAEAAACAG5okAAAAAHBDkwQAAAAAbmiSAAAAAMANTRIAAAAAuKFJAgAAAAA3NEkAAAAA4IYmCQAAAADcBORmZqfTqZUrV+rrr7/WH3/8odOnTysqKkqNGjVShw4dFBMTU1B5AgAAAEChyNGRpDNnzuipp55STEyMOnfurMWLF+vYsWOy2+367bffNHbsWMXFxalz5876/vvvCzpnAAAAACgwOTqSdNlll6lFixZ688031bFjRwUGBnrN88cff2ju3Lnq27evHnnkEd155535niwAAAAAFLQcNUlLlixR7dq1zztPbGysRo0apYceeki7du3Kl+QAAAAAoLDl6HS7CzVI7gIDAxUfH3/RCQEAAACAP+Xqxg3u0tPT9frrr2vFihVyOBxq2bKlhgwZopCQkPzMDwAAAAAK1UU3Sffff7+2bdumHj16KC0tTbNnz9batWv13nvv5Wd+AAAAAFCoctwkLViwQN27d3c9XrJkibZu3Sq73S5J6tSpk6688sr8zxAAAAAAClGO/5js22+/rW7dumnPnj2SpMaNG+uee+7R559/rk8//VQjR45Us2bNCixRAAAAACgMOW6SPv30U/Xr109t2rTRq6++qjfeeEPh4eF65JFH9NhjjykmJkZz584tyFwBAAAAoMDl6pqkPn36qFOnTho5cqQ6deqkadOm6YUXXiio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3TodOXJE69ev/8d+gfynePzxx7Vw4UJt2LDB36nkm71796p06dKux2PHjlVYWJi2bt2qkiVLZhv7p+FIEnIlNTX1op8bFRWlEiVK5GM2AHB+ERERioyM9HcauEScOnVKV199tSZOnOjvVFDEpKWl+TuFHKtYsaLHbcd///13XX311YqNjVXZsmWzjf3T0CThvNq0aaOhQ4dq2LBhKleunDp16qQXX3xR9erVU1hYmGJiYnTvvffq5MmTkqQVK1Zo0KBBSk5OlmVZsixLjz/+uCTvw727du3SjTfeqJIlSyo8PFy9e/fW/v37/TBKFGWLFi1SRESE3n33XX+ngmLI/XS7Dz/8UPXq1VNoaKjKli2rDh066NSpU/5NEMXKrbfeqjFjxqhDhw7+TgU55HQ6NX78eMXFxSk0NFQNGjTQhx9+KOncdxbLsrR8+XI1bdpUJUqU0FVXXaWtW7dKOndmzLhx47Rx40bXd5qZM2dKOnfK2tSpU/Wvf/1LYWFhevrppyVJU6dOVXx8vIKCglSrVi298847HvlkPO/6669XaGioqlev7spHktq1a6ehQ4d6POfgwYMKCgrS8uXLLzjeAwcOqGvXrgoNDVVcXJzPfaf76XaWZWndunV64oknXN/ZfMX+iWiScEGzZs1SUFCQvv32W02bNk02m02vvPKKNm3apFmzZunLL7/UyJEjJUlXXXWVJk2apPDwcO3du1d79+7VQw895LVMp9OpG2+8UUeOHNHKlSu1dOlS7dixQ3369Cns4aEImzt3rvr166d3331X/fv393c6KMb27t2rfv36afDgwfr111+1YsUK9ejRg1PxgEvc+PHjNXv2bE2bNk2bNm3S8OHDdcstt2jlypWueR555BG98MILWrt2rQICAjR48GBJUp8+ffTggw+qbt26ru807t9THn/8cXXv3l2//PKLBg8erAULFuiBBx7Qgw8+qMTERN19990aNGiQvvrqK4+cHnvsMfXs2VMbN25U//791bdvX/3666+SpDvuuENz585VSkqKa/45c+aoSpUqateu3QXHO3DgQP3555/66quv9OGHH2rKlCk6cOBAtvPv3btXdevW1YMPPuj6zuYr9k/ENUm4oJo1a+rZZ591Pa5Vq5br/9WqVdNTTz2le+65R1OmTFFQUJAiIiJkWZYqVqyY7TKXL1+uX375RUlJSYqJ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",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "BATCH_SIZE = 64\n",
    "LEARNING_RATE = 0.01\n",
    "LOCAL_EPOCHS = 5\n",
    "NUM_OF_CLIENTS = 10\n",
    "COMM_ROUND = 30\n",
    "ALPHA = 0.5  \n",
    "FRAC = 0.1   \n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "def run_all_measures():\n",
    "    measures = [\"ratio\", \"kl\", \"js\", \"l1\", \"entropy_diff\"]\n",
    "    \n",
    "    results = {\n",
    "        \"measure\": [],\n",
    "        \"mean_local_acc\": [],\n",
    "        \"final_server_acc\": []\n",
    "    }\n",
    "    \n",
    "    for m in measures:\n",
    "        client_accs, server_acc, _ = run_fedchill_het(metric=m)\n",
    "        \n",
    "        \n",
    "        mean_local = np.mean(client_accs)\n",
    "        \n",
    "        results[\"measure\"].append(m)\n",
    "        results[\"mean_local_acc\"].append(mean_local)\n",
    "        results[\"final_server_acc\"].append(server_acc)\n",
    "    \n",
    "    return results\n",
    "\n",
    "\n",
    "def plot_results(results):\n",
    "    measures = results[\"measure\"]\n",
    "    x = np.arange(len(measures))\n",
    "    \n",
    "    plt.figure(figsize=(10,6))\n",
    "    \n",
    "    \n",
    "    plt.bar(x - 0.2, results[\"mean_local_acc\"], width=0.4, label=\"Mean Local Accuracy\")\n",
    "    \n",
    "    \n",
    "    plt.bar(x + 0.2, results[\"final_server_acc\"], width=0.4, label=\"Final Server Accuracy\")\n",
    "    \n",
    "    plt.xticks(x, measures)\n",
    "    plt.ylabel(\"Accuracy (%)\")\n",
    "    plt.title(\"Comparison of Heterogeneity Measures on Local & Server Accuracy\")\n",
    "    plt.legend()\n",
    "    plt.grid(axis=\"y\", linestyle=\"--\", alpha=0.7)\n",
    "    plt.show()\n",
    "\n",
    "\n",
    "\n",
    "results = run_all_measures()\n",
    "plot_results(results)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T11:21:55.809577Z",
     "iopub.status.busy": "2025-09-16T11:21:55.809224Z",
     "iopub.status.idle": "2025-09-16T11:21:55.813824Z",
     "shell.execute_reply": "2025-09-16T11:21:55.813212Z",
     "shell.execute_reply.started": "2025-09-16T11:21:55.809546Z"
    },
    "trusted": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'measure': ['ratio', 'kl', 'js', 'l1', 'entropy_diff'], 'mean_local_acc': [68.73697916666666, 68.22395833333334, 68.61197916666666, 67.15625, 67.69270833333333], 'final_server_acc': [50.02, 47.73, 55.61, 51.75, 48.64]}\n"
     ]
    }
   ],
   "source": [
    "print(results)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Clients 10, Frac 0.1, Rounds 30, S = -0.5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T12:35:07.800041Z",
     "iopub.status.busy": "2025-09-16T12:35:07.799311Z",
     "iopub.status.idle": "2025-09-16T12:35:07.811238Z",
     "shell.execute_reply": "2025-09-16T12:35:07.810482Z",
     "shell.execute_reply.started": "2025-09-16T12:35:07.800013Z"
    },
    "trusted": true
   },
   "outputs": [],
   "source": [
    "class AdaptiveClientHyperparams:\n",
    "    def __init__(self, initial_lr=0.01, min_lr=0.0001, max_lr=0.05, \n",
    "                 initial_temp=0.5, min_temp=0.05, max_temp=1.0,\n",
    "                 patience=2, lr_decay_factor=0.7, temp_adjust_factor=0.8):\n",
    "        \n",
    "        self.lr = initial_lr\n",
    "        self.min_lr = min_lr\n",
    "        self.max_lr = max_lr\n",
    "        self.lr_decay_factor = lr_decay_factor\n",
    "        \n",
    "        \n",
    "        self.temp = initial_temp\n",
    "        self.min_temp = min_temp\n",
    "        self.max_temp = max_temp\n",
    "        self.temp_adjust_factor = temp_adjust_factor\n",
    "        \n",
    "        \n",
    "        self.patience = patience\n",
    "        self.performance_history = []\n",
    "        self.stagnation_count = 0\n",
    "        self.improvement_count = 0\n",
    "        self.het_score = 0.0    \n",
    "        \n",
    "    def update_temp_from_heterogeneity(self, het_score):\n",
    "        \"\"\"Initialize temperature based on client's heterogeneity score\"\"\"\n",
    "        self.het_score = het_score\n",
    "        \n",
    "        \n",
    "        self.temp = self.max_temp * math.exp(-0.5 * het_score) \n",
    "        self.temp = min(max(self.temp, self.min_temp), self.max_temp)\n",
    "        \n",
    "    def record_performance(self, accuracy):\n",
    "        \"\"\"Record performance and adjust hyperparameters if needed\"\"\"\n",
    "        self.performance_history.append(accuracy)\n",
    "        \n",
    "        \n",
    "        if len(self.performance_history) >= 3:\n",
    "            \n",
    "            if self.performance_history[-1] <= self.performance_history[-2]:\n",
    "                self.stagnation_count += 1\n",
    "                self.improvement_count = 0\n",
    "            else:\n",
    "                self.improvement_count += 1\n",
    "                self.stagnation_count = 0\n",
    "                \n",
    "            \n",
    "            if self.stagnation_count >= self.patience:\n",
    "                \n",
    "                if self.temp > self.min_temp * 1.1:  \n",
    "                    self.temp *= self.temp_adjust_factor\n",
    "                    self.temp = max(self.temp, self.min_temp)\n",
    "                    self.stagnation_count = 0\n",
    "                    return True, \"decrease-temp\", self.temp\n",
    "        \n",
    "        return False, \"unchanged\", None\n",
    "        \n",
    "    def get_lr(self):\n",
    "        return self.lr\n",
    "        \n",
    "    def get_temp(self):\n",
    "        return self.temp\n",
    "    \n",
    "\n",
    "def calculate_heterogeneity_score(client_distribution, global_distribution):\n",
    "    \"\"\"Calculate how different client's class distribution is from global distribution\"\"\"\n",
    "    score = 0\n",
    "    for cls in range(10):\n",
    "        client_prob = client_distribution.get(cls, 0)\n",
    "        global_prob = global_distribution.get(cls, 1/10)\n",
    "        if client_prob > 0 and global_prob > 0:\n",
    "            ratio = client_prob / global_prob\n",
    "            score += abs(ratio - 1)\n",
    "    \n",
    "    \n",
    "    score = min(score / 10, 1)\n",
    "    return score\n",
    "\n",
    "def calculate_global_distribution(client_class_distributions):\n",
    "    \"\"\"Calculate approximate global class distribution from client distributions\"\"\"\n",
    "    global_dist = {}\n",
    "    n_clients = len(client_class_distributions)\n",
    "    \n",
    "    for cls in range(10):\n",
    "        global_dist[cls] = sum(dist.get(cls, 0) for dist in client_class_distributions) / n_clients\n",
    "    \n",
    "    for cls in range(10):\n",
    "        if global_dist[cls] < 0.01:\n",
    "            global_dist[cls] = 0.01\n",
    "    \n",
    "    total = sum(global_dist.values())\n",
    "    global_dist = {k: v/total for k, v in global_dist.items()}\n",
    "    \n",
    "    return global_dist"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T12:41:31.516505Z",
     "iopub.status.busy": "2025-09-16T12:41:31.516181Z",
     "iopub.status.idle": "2025-09-16T13:48:35.799265Z",
     "shell.execute_reply": "2025-09-16T13:48:35.798391Z",
     "shell.execute_reply.started": "2025-09-16T12:41:31.516467Z"
    },
    "trusted": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Running FedChill with S = 0.5\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = ratio\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 0.9007, Initial temp: 0.6374\n",
      "Client 1 - Het Score: 0.7295, Initial temp: 0.6944\n",
      "Client 2 - Het Score: 0.8682, Initial temp: 0.6478\n",
      "Client 3 - Het Score: 0.8552, Initial temp: 0.6521\n",
      "Client 4 - Het Score: 0.5698, Initial temp: 0.7521\n",
      "Client 5 - Het Score: 0.5956, Initial temp: 0.7424\n",
      "Client 6 - Het Score: 0.7451, Initial temp: 0.6890\n",
      "Client 7 - Het Score: 0.6556, Initial temp: 0.7205\n",
      "Client 8 - Het Score: 1.0000, Initial temp: 0.6065\n",
      "Client 9 - Het Score: 1.0000, Initial temp: 0.6065\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.5099, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1365\n",
      "Epoch [2/5], Loss: 0.0335\n",
      "Epoch [3/5], Loss: 0.0185\n",
      "Epoch [4/5], Loss: 0.0147\n",
      "Epoch [5/5], Loss: 0.0121\n",
      "Client 0 Private Data Validation Accuracy: 64.38%\n",
      "\n",
      "--- Client 1 Local Training with T=0.4444, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.6590\n",
      "Epoch [2/5], Loss: 0.1848\n",
      "Epoch [3/5], Loss: 0.1070\n",
      "Epoch [4/5], Loss: 0.0570\n",
      "Epoch [5/5], Loss: 0.0345\n",
      "Client 1 Private Data Validation Accuracy: 65.62%\n",
      "\n",
      "--- Client 2 Local Training with T=0.3317, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3505\n",
      "Epoch [2/5], Loss: 0.1603\n",
      "Epoch [3/5], Loss: 0.0639\n",
      "Epoch [4/5], Loss: 0.0960\n",
      "Epoch [5/5], Loss: 0.0626\n",
      "Client 2 Private Data Validation Accuracy: 72.40%\n",
      "\n",
      "--- Client 3 Local Training with T=0.2137, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3326\n",
      "Epoch [2/5], Loss: 0.1390\n",
      "Epoch [3/5], Loss: 0.0561\n",
      "Epoch [4/5], Loss: 0.0507\n",
      "Epoch [5/5], Loss: 0.0229\n",
      "Client 3 Private Data Validation Accuracy: 93.23%\n",
      "\n",
      "--- Client 4 Local Training with T=0.3081, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1106\n",
      "Epoch [2/5], Loss: 0.0731\n",
      "Epoch [3/5], Loss: 0.0222\n",
      "Epoch [4/5], Loss: 0.0120\n",
      "Epoch [5/5], Loss: 0.0091\n",
      "Client 4 Private Data Validation Accuracy: 61.98%\n",
      "\n",
      "--- Client 5 Local Training with T=0.3801, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2664\n",
      "Epoch [2/5], Loss: 0.0912\n",
      "Epoch [3/5], Loss: 0.0736\n",
      "Epoch [4/5], Loss: 0.0310\n",
      "Epoch [5/5], Loss: 0.0290\n",
      "Client 5 Private Data Validation Accuracy: 58.59%\n",
      "\n",
      "--- Client 6 Local Training with T=0.2822, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.1106\n",
      "Epoch [2/5], Loss: 0.3581\n",
      "Epoch [3/5], Loss: 0.1683\n",
      "Epoch [4/5], Loss: 0.0285\n",
      "Epoch [5/5], Loss: 0.0142\n",
      "Client 6 Private Data Validation Accuracy: 68.75%\n",
      "\n",
      "--- Client 7 Local Training with T=0.1889, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2500\n",
      "Epoch [2/5], Loss: 1.5000\n",
      "Epoch [3/5], Loss: 1.3834\n",
      "Epoch [4/5], Loss: 0.6677\n",
      "Epoch [5/5], Loss: 0.3097\n",
      "Client 7 Private Data Validation Accuracy: 50.00%\n",
      "\n",
      "--- Client 8 Local Training with T=0.3105, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1757\n",
      "Epoch [2/5], Loss: 0.0285\n",
      "Epoch [3/5], Loss: 0.0230\n",
      "Epoch [4/5], Loss: 0.0311\n",
      "Epoch [5/5], Loss: 0.0175\n",
      "Client 8 Private Data Validation Accuracy: 79.30%\n",
      "\n",
      "--- Client 9 Local Training with T=0.1987, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2845\n",
      "Epoch [2/5], Loss: 0.3151\n",
      "Epoch [3/5], Loss: 0.3642\n",
      "Epoch [4/5], Loss: 0.1383\n",
      "Epoch [5/5], Loss: 0.0613\n",
      "Client 9 Private Data Validation Accuracy: 60.16%\n",
      "Client 9 - decrease-temp to 0.158998\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 54.50%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 64.38%\n",
      "Client 1 Final Private Data Validation Accuracy: 65.62%\n",
      "Client 2 Final Private Data Validation Accuracy: 72.40%\n",
      "Client 3 Final Private Data Validation Accuracy: 93.23%\n",
      "Client 4 Final Private Data Validation Accuracy: 63.02%\n",
      "Client 5 Final Private Data Validation Accuracy: 54.69%\n",
      "Client 6 Final Private Data Validation Accuracy: 65.62%\n",
      "Client 7 Final Private Data Validation Accuracy: 49.48%\n",
      "Client 8 Final Private Data Validation Accuracy: 78.91%\n",
      "Client 9 Final Private Data Validation Accuracy: 65.62%\n",
      "\n",
      "Final Server Test Accuracy: 54.50%\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = kl\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 1.0227, Initial temp: 0.5997\n",
      "Client 1 - Het Score: 0.6109, Initial temp: 0.7368\n",
      "Client 2 - Het Score: 1.4072, Initial temp: 0.4948\n",
      "Client 3 - Het Score: 1.6845, Initial temp: 0.4307\n",
      "Client 4 - Het Score: 0.5174, Initial temp: 0.7721\n",
      "Client 5 - Het Score: 0.4435, Initial temp: 0.8011\n",
      "Client 6 - Het Score: 0.7640, Initial temp: 0.6825\n",
      "Client 7 - Het Score: 0.3035, Initial temp: 0.8592\n",
      "Client 8 - Het Score: 1.1237, Initial temp: 0.5701\n",
      "Client 9 - Het Score: 0.7487, Initial temp: 0.6878\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.3070, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.0825\n",
      "Epoch [2/5], Loss: 0.0401\n",
      "Epoch [3/5], Loss: 0.0225\n",
      "Epoch [4/5], Loss: 0.0266\n",
      "Epoch [5/5], Loss: 0.0149\n",
      "Client 0 Private Data Validation Accuracy: 60.31%\n",
      "\n",
      "--- Client 1 Local Training with T=0.3772, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.7949\n",
      "Epoch [2/5], Loss: 0.2236\n",
      "Epoch [3/5], Loss: 0.0914\n",
      "Epoch [4/5], Loss: 0.0637\n",
      "Epoch [5/5], Loss: 0.0444\n",
      "Client 1 Private Data Validation Accuracy: 75.00%\n",
      "\n",
      "--- Client 2 Local Training with T=0.2027, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.4644\n",
      "Epoch [2/5], Loss: 0.3538\n",
      "Epoch [3/5], Loss: 0.5569\n",
      "Epoch [4/5], Loss: 0.2289\n",
      "Epoch [5/5], Loss: 0.1484\n",
      "Client 2 Private Data Validation Accuracy: 65.62%\n",
      "\n",
      "--- Client 3 Local Training with T=0.1764, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3982\n",
      "Epoch [2/5], Loss: 0.1829\n",
      "Epoch [3/5], Loss: 0.0339\n",
      "Epoch [4/5], Loss: 0.0122\n",
      "Epoch [5/5], Loss: 0.0016\n",
      "Client 3 Private Data Validation Accuracy: 92.19%\n",
      "\n",
      "--- Client 4 Local Training with T=0.4941, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1579\n",
      "Epoch [2/5], Loss: 0.0464\n",
      "Epoch [3/5], Loss: 0.0237\n",
      "Epoch [4/5], Loss: 0.0181\n",
      "Epoch [5/5], Loss: 0.0142\n",
      "Client 4 Private Data Validation Accuracy: 64.58%\n",
      "Client 4 - decrease-temp to 0.395302\n",
      "\n",
      "--- Client 5 Local Training with T=0.5127, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3694\n",
      "Epoch [2/5], Loss: 0.1491\n",
      "Epoch [3/5], Loss: 0.0677\n",
      "Epoch [4/5], Loss: 0.0361\n",
      "Epoch [5/5], Loss: 0.0376\n",
      "Client 5 Private Data Validation Accuracy: 56.25%\n",
      "\n",
      "--- Client 6 Local Training with T=0.1145, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 7.0069\n",
      "Epoch [2/5], Loss: 8.7305\n",
      "Epoch [3/5], Loss: 6.3379\n",
      "Epoch [4/5], Loss: 3.2027\n",
      "Epoch [5/5], Loss: 1.1215\n",
      "Client 6 Private Data Validation Accuracy: 6.25%\n",
      "\n",
      "--- Client 7 Local Training with T=0.2815, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1677\n",
      "Epoch [2/5], Loss: 0.0914\n",
      "Epoch [3/5], Loss: 0.0611\n",
      "Epoch [4/5], Loss: 0.0531\n",
      "Epoch [5/5], Loss: 0.0298\n",
      "Client 7 Private Data Validation Accuracy: 66.67%\n",
      "\n",
      "--- Client 8 Local Training with T=0.2919, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1446\n",
      "Epoch [2/5], Loss: 0.0415\n",
      "Epoch [3/5], Loss: 0.0201\n",
      "Epoch [4/5], Loss: 0.0127\n",
      "Epoch [5/5], Loss: 0.0095\n",
      "Client 8 Private Data Validation Accuracy: 78.91%\n",
      "\n",
      "--- Client 9 Local Training with T=0.3521, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2095\n",
      "Epoch [2/5], Loss: 0.0517\n",
      "Epoch [3/5], Loss: 0.0326\n",
      "Epoch [4/5], Loss: 0.0342\n",
      "Epoch [5/5], Loss: 0.0132\n",
      "Client 9 Private Data Validation Accuracy: 72.66%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 54.26%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 60.31%\n",
      "Client 1 Final Private Data Validation Accuracy: 73.44%\n",
      "Client 2 Final Private Data Validation Accuracy: 65.62%\n",
      "Client 3 Final Private Data Validation Accuracy: 92.71%\n",
      "Client 4 Final Private Data Validation Accuracy: 64.06%\n",
      "Client 5 Final Private Data Validation Accuracy: 53.12%\n",
      "Client 6 Final Private Data Validation Accuracy: 6.25%\n",
      "Client 7 Final Private Data Validation Accuracy: 67.71%\n",
      "Client 8 Final Private Data Validation Accuracy: 78.91%\n",
      "Client 9 Final Private Data Validation Accuracy: 74.22%\n",
      "\n",
      "Final Server Test Accuracy: 54.26%\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = js\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 0.2739, Initial temp: 0.8720\n",
      "Client 1 - Het Score: 0.1681, Initial temp: 0.9194\n",
      "Client 2 - Het Score: 0.3461, Initial temp: 0.8411\n",
      "Client 3 - Het Score: 0.4343, Initial temp: 0.8048\n",
      "Client 4 - Het Score: 0.1533, Initial temp: 0.9262\n",
      "Client 5 - Het Score: 0.1192, Initial temp: 0.9421\n",
      "Client 6 - Het Score: 0.1991, Initial temp: 0.9052\n",
      "Client 7 - Het Score: 0.0819, Initial temp: 0.9599\n",
      "Client 8 - Het Score: 0.2538, Initial temp: 0.8808\n",
      "Client 9 - Het Score: 0.1906, Initial temp: 0.9091\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.5581, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1138\n",
      "Epoch [2/5], Loss: 0.0260\n",
      "Epoch [3/5], Loss: 0.0196\n",
      "Epoch [4/5], Loss: 0.0121\n",
      "Epoch [5/5], Loss: 0.0093\n",
      "Client 0 Private Data Validation Accuracy: 59.38%\n",
      "\n",
      "--- Client 1 Local Training with T=0.4707, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.8861\n",
      "Epoch [2/5], Loss: 0.3159\n",
      "Epoch [3/5], Loss: 0.1499\n",
      "Epoch [4/5], Loss: 0.0752\n",
      "Epoch [5/5], Loss: 0.0373\n",
      "Client 1 Private Data Validation Accuracy: 64.06%\n",
      "\n",
      "--- Client 2 Local Training with T=0.3445, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.4274\n",
      "Epoch [2/5], Loss: 0.1342\n",
      "Epoch [3/5], Loss: 0.0758\n",
      "Epoch [4/5], Loss: 0.1414\n",
      "Epoch [5/5], Loss: 0.0919\n",
      "Client 2 Private Data Validation Accuracy: 69.79%\n",
      "\n",
      "--- Client 3 Local Training with T=0.4121, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2601\n",
      "Epoch [2/5], Loss: 0.0320\n",
      "Epoch [3/5], Loss: 0.0120\n",
      "Epoch [4/5], Loss: 0.0123\n",
      "Epoch [5/5], Loss: 0.0055\n",
      "Client 3 Private Data Validation Accuracy: 90.62%\n",
      "\n",
      "--- Client 4 Local Training with T=0.3035, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.0882\n",
      "Epoch [2/5], Loss: 0.0724\n",
      "Epoch [3/5], Loss: 0.0262\n",
      "Epoch [4/5], Loss: 0.0159\n",
      "Epoch [5/5], Loss: 0.0093\n",
      "Client 4 Private Data Validation Accuracy: 60.42%\n",
      "\n",
      "--- Client 5 Local Training with T=0.3087, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3973\n",
      "Epoch [2/5], Loss: 0.5173\n",
      "Epoch [3/5], Loss: 0.4061\n",
      "Epoch [4/5], Loss: 0.1620\n",
      "Epoch [5/5], Loss: 0.1172\n",
      "Client 5 Private Data Validation Accuracy: 57.81%\n",
      "\n",
      "--- Client 6 Local Training with T=0.3708, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.0717\n",
      "Epoch [2/5], Loss: 0.4148\n",
      "Epoch [3/5], Loss: 0.1314\n",
      "Epoch [4/5], Loss: 0.0289\n",
      "Epoch [5/5], Loss: 0.0256\n",
      "Client 6 Private Data Validation Accuracy: 65.62%\n",
      "Client 6 - decrease-temp to 0.296627\n",
      "\n",
      "--- Client 7 Local Training with T=0.2516, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.6692\n",
      "Epoch [2/5], Loss: 1.2883\n",
      "Epoch [3/5], Loss: 0.5707\n",
      "Epoch [4/5], Loss: 0.3867\n",
      "Epoch [5/5], Loss: 0.1307\n",
      "Client 7 Private Data Validation Accuracy: 59.90%\n",
      "\n",
      "--- Client 8 Local Training with T=0.2886, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.4897\n",
      "Epoch [2/5], Loss: 0.4244\n",
      "Epoch [3/5], Loss: 0.1854\n",
      "Epoch [4/5], Loss: 0.1289\n",
      "Epoch [5/5], Loss: 0.0737\n",
      "Client 8 Private Data Validation Accuracy: 80.86%\n",
      "\n",
      "--- Client 9 Local Training with T=0.3724, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2475\n",
      "Epoch [2/5], Loss: 0.0590\n",
      "Epoch [3/5], Loss: 0.0310\n",
      "Epoch [4/5], Loss: 0.0218\n",
      "Epoch [5/5], Loss: 0.0108\n",
      "Client 9 Private Data Validation Accuracy: 77.34%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 51.29%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 59.38%\n",
      "Client 1 Final Private Data Validation Accuracy: 67.19%\n",
      "Client 2 Final Private Data Validation Accuracy: 68.75%\n",
      "Client 3 Final Private Data Validation Accuracy: 90.10%\n",
      "Client 4 Final Private Data Validation Accuracy: 58.85%\n",
      "Client 5 Final Private Data Validation Accuracy: 54.69%\n",
      "Client 6 Final Private Data Validation Accuracy: 64.06%\n",
      "Client 7 Final Private Data Validation Accuracy: 62.50%\n",
      "Client 8 Final Private Data Validation Accuracy: 80.47%\n",
      "Client 9 Final Private Data Validation Accuracy: 76.56%\n",
      "\n",
      "Final Server Test Accuracy: 51.29%\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = entropy_diff\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 0.8527, Initial temp: 0.6529\n",
      "Client 1 - Het Score: 0.7037, Initial temp: 0.7034\n",
      "Client 2 - Het Score: 1.3864, Initial temp: 0.5000\n",
      "Client 3 - Het Score: 1.6167, Initial temp: 0.4456\n",
      "Client 4 - Het Score: 0.5006, Initial temp: 0.7786\n",
      "Client 5 - Het Score: 0.4251, Initial temp: 0.8085\n",
      "Client 6 - Het Score: 0.9316, Initial temp: 0.6276\n",
      "Client 7 - Het Score: 0.3201, Initial temp: 0.8521\n",
      "Client 8 - Het Score: 1.0807, Initial temp: 0.5825\n",
      "Client 9 - Het Score: 0.8083, Initial temp: 0.6676\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.2139, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1812\n",
      "Epoch [2/5], Loss: 1.8121\n",
      "Epoch [3/5], Loss: 0.9024\n",
      "Epoch [4/5], Loss: 0.3140\n",
      "Epoch [5/5], Loss: 0.1557\n",
      "Client 0 Private Data Validation Accuracy: 52.19%\n",
      "\n",
      "--- Client 1 Local Training with T=0.3601, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.6761\n",
      "Epoch [2/5], Loss: 0.2200\n",
      "Epoch [3/5], Loss: 0.0872\n",
      "Epoch [4/5], Loss: 0.0450\n",
      "Epoch [5/5], Loss: 0.0330\n",
      "Client 1 Private Data Validation Accuracy: 68.75%\n",
      "\n",
      "--- Client 2 Local Training with T=0.3200, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3798\n",
      "Epoch [2/5], Loss: 0.0467\n",
      "Epoch [3/5], Loss: 0.0498\n",
      "Epoch [4/5], Loss: 0.0599\n",
      "Epoch [5/5], Loss: 0.0321\n",
      "Client 2 Private Data Validation Accuracy: 73.44%\n",
      "\n",
      "--- Client 3 Local Training with T=0.2281, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3276\n",
      "Epoch [2/5], Loss: 0.0524\n",
      "Epoch [3/5], Loss: 0.0137\n",
      "Epoch [4/5], Loss: 0.0111\n",
      "Epoch [5/5], Loss: 0.0021\n",
      "Client 3 Private Data Validation Accuracy: 90.10%\n",
      "\n",
      "--- Client 4 Local Training with T=0.2551, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.0607\n",
      "Epoch [2/5], Loss: 0.0202\n",
      "Epoch [3/5], Loss: 0.0353\n",
      "Epoch [4/5], Loss: 0.0328\n",
      "Epoch [5/5], Loss: 0.0233\n",
      "Client 4 Private Data Validation Accuracy: 59.90%\n",
      "\n",
      "--- Client 5 Local Training with T=0.4140, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3473\n",
      "Epoch [2/5], Loss: 0.1197\n",
      "Epoch [3/5], Loss: 0.0583\n",
      "Epoch [4/5], Loss: 0.0360\n",
      "Epoch [5/5], Loss: 0.0355\n",
      "Client 5 Private Data Validation Accuracy: 59.38%\n",
      "\n",
      "--- Client 6 Local Training with T=0.1645, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 2.6388\n",
      "Epoch [2/5], Loss: 2.7832\n",
      "Epoch [3/5], Loss: 1.3064\n",
      "Epoch [4/5], Loss: 0.2136\n",
      "Epoch [5/5], Loss: 0.0594\n",
      "Client 6 Private Data Validation Accuracy: 60.94%\n",
      "\n",
      "--- Client 7 Local Training with T=0.2792, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1390\n",
      "Epoch [2/5], Loss: 0.0563\n",
      "Epoch [3/5], Loss: 0.0407\n",
      "Epoch [4/5], Loss: 0.0235\n",
      "Epoch [5/5], Loss: 0.0169\n",
      "Client 7 Private Data Validation Accuracy: 65.62%\n",
      "\n",
      "--- Client 8 Local Training with T=0.2386, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1554\n",
      "Epoch [2/5], Loss: 0.0328\n",
      "Epoch [3/5], Loss: 0.0194\n",
      "Epoch [4/5], Loss: 0.0063\n",
      "Epoch [5/5], Loss: 0.0294\n",
      "Client 8 Private Data Validation Accuracy: 79.30%\n",
      "\n",
      "--- Client 9 Local Training with T=0.2187, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2286\n",
      "Epoch [2/5], Loss: 0.1485\n",
      "Epoch [3/5], Loss: 0.2687\n",
      "Epoch [4/5], Loss: 0.1222\n",
      "Epoch [5/5], Loss: 0.0865\n",
      "Client 9 Private Data Validation Accuracy: 68.75%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 50.73%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 52.19%\n",
      "Client 1 Final Private Data Validation Accuracy: 70.31%\n",
      "Client 2 Final Private Data Validation Accuracy: 72.40%\n",
      "Client 3 Final Private Data Validation Accuracy: 89.58%\n",
      "Client 4 Final Private Data Validation Accuracy: 61.98%\n",
      "Client 5 Final Private Data Validation Accuracy: 57.03%\n",
      "Client 6 Final Private Data Validation Accuracy: 59.38%\n",
      "Client 7 Final Private Data Validation Accuracy: 64.06%\n",
      "Client 8 Final Private Data Validation Accuracy: 78.91%\n",
      "Client 9 Final Private Data Validation Accuracy: 71.09%\n",
      "\n",
      "Final Server Test Accuracy: 50.73%\n"
     ]
    },
    {
     "data": {
      "image/png": "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7dq189qpmBuvdT4DBgwwcXFxns9n0aJFRpJXcZWenm4qVapkKlSo4LVTxxjvz7VZs2amaNGiXn97z10np0VSdu3xZy7wpz/+3XkPhQ+n26HAOHbsmCSpaNGifq3/v//9T5L02GOPecUHDRokST7XLtWoUUNNmjTxPL7qqqskSS1btlT58uV94jt27PB5TftdnTIO76empmrJkiWeeGhoqOf///rrLyUnJ+vaa6/1OTVOkpo3b64aNWpcoKVnr+tZtWqV9u7dm+Xy1atX6+DBg3rwwQe9zsFu3769qlevnuV1XPfff7/X42uvvTbLNtstWbJEqampGjhwoNcF/vfcc48iIiL+9vViH330kRYvXuzzr1SpUl7rzZ49W5GRkWrTpo0OHz7s+Ve/fn0VKVJEy5Ytu+BrzZ49W9dee62KFSvmtY3WrVvL5XJpxYoVXut36dJF0dHRnsf79u3TunXr1Lt3bxUvXtwTr1Onjtq0aePpny6XS0uWLFHHjh1VtmxZz3pVqlTRjTfe6PUac+fOldvtVteuXb1yKl26tKpWrerTriJFiujOO+/0PA4KClKjRo28PsfZs2crISFB1atX99pmy5YtJcmv9+pcDodDd9xxhz755BMdP37cE58xY4auvvpqVapUye9t3X777Vq+fLn279+vL774Qvv378/2VLucfGb+jMOM6+U+/vhjn9P1LlZwcLDPqVH+fgYXm8/FzIXXXnut53F0dLSqVat2wfHvby5Op1MPP/ywTy7GGC1cuFCSNH/+fLndbj377LNec4kkr1Op7J/j8ePHdfjwYV177bU6deqUNm/enOP8Bg8erIULF+rnn3/WVVddpXbt2mndunWe5fv27ZNlWV7XK2Vl+PDhSkpKUr169fT555/rqaeeUv369XXllVd6nW6b0/GX1d+Eli1bqmTJkpo1a5Yn9tdff2nx4sXq1q1brr5WdtLT0zVr1ix169bN8/m0bNlSMTExmjFjhme9tWvXaufOnRo4cKDP9agZzzt06JBWrFihu+++2+tvr32di5Fde/yZC/zpj7k576Fw4MYNKDAiIiIkyWvyOZ/ffvtNDofD585ppUuXVlRUlH777Tev+LmTcWRkpCQpLi4uy7j92hbp7ARZuXJlr9gVV1whSV7XK/z3v//VCy+8oHXr1nmdB57V5O/vpPryyy+rV69eiouLU/36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/PPPa+zYsSpdunS26/n7mfk7Dv/880+f7WR8QcoYRxlf+uzXO7lcrhz9oLW/n4E/+WQlP+bC8+Xicrm85kpJev3112VZlufId8eOHeVwOPTcc8/5HBHK6PsZR7zsYyE1NVXjx4+/qNyuueYaBQcHa+TIkTp16pQnHh8frzFjxmj37t2KjIxU8+bNz7udU6dOaeXKlVkuyxjTGafh/p3xZ+dwOHTrrbfq008/1QcffKD09HSvU+1y87XONW/ePJ08eVL9+/fXrbfe6vPvpptu0kcffaSUlBRdeeWVqlSpksaMGeMzb2R8jtHR0WrWrJnef/997d69O8t1pLOfS3Jystfpw/v27dO8efP8zt3fucCf/pght+Y9FHwcSUKBEh8fr6SkJHXr1k0JCQnq2bOnatWqpdTUVH377beaPXu253dcEhMT1atXL73zzjue0zK+//57TZ06VR07dsz1Q/8hISH67LPP1KtXL1111VVauHChFixYoCeffNJzelf79u01evRo3XDDDbr99tt18OBBjRs3TlWqVPGa6HPi+PHjio2N1a233qrExEQVKVJES5Ys0Q8//KDXXntN0tlrVUaNGqU+ffqoefPm6tGjhw4cOKA33nhDFStW1KOPPpor70F0dLSGDBmi4cOH64YbbtDNN9+sLVu2aPz48WrYsGGe/bFo3ry57rvvPo0YMULr1q3T9ddfr8DAQG3btk2zZ8/WG2+84dnjWb9+fU2YMEEvvPCCqlSpopiYGLVs2VL/+te/9Mknn+imm25S7969Vb9+fZ08eVI///yz5syZo127dmV5fZTdK6+8ohtvvFFNmjRR3759dfr0ab311luKjIz0Om1s2LBhWrRokZo2baoHHnjA8yWyVq1aXheNx8fH64UXXtCQIUO0a9cudezYUUWLFtXOnTs1b9483XvvvV6/ieKPu+66S//5z390//33a9myZWratKlcLpc2b96s//znP/r888/VoEGDLJ9bt25dOZ1OjRo1SsnJyQoODvZcrC2d7Q833HCDZs+eraioqIv+Mu5wOPT0009fcD1/PzN/x+Fzzz2nFStWqH379qpQoYIOHjyo8ePHKzY2Vtdcc40kqWbNmmrcuLGGDBmiP//8U8WLF9fMmTMvWETb+fsZ+JNPVvJ6Lly9erVeeOEFn3iLFi3UoUMHXXfddXrqqae0a9cuJSYmatGiRfr44481cOBAT9FZpUoVPfXUU3r++ed17bXXqnPnzgoODtYPP/ygsmXLasSIEbr66qtVrFgx9erVSw8//LAsy9IHH3xw0acFRkdHa8SIEXrsscdUu3Zt3X333SpdurRWr16tqVOnqnHjxvrxxx916623auHChT7XAGY4deqUrr76ajVu3Fg33HCD4uLidPToUc2fP19fffWVOnbsqHr16kn6e+PvXN26ddNbb72loUOHqnbt2l5HDXP7texmzJihEiVK6Oqrr85y+c0336x3331XCxYsUOfOnTVhwgR16NBBdevWVZ8+fVSmTBlt3rxZGzdu9OzgePPNN3XNNdfoyiuv1L333qtKlSpp165dWrBggWdO7N69u5544gl16tRJDz/8sE6dOqUJEyboiiuuyPJGSFnxdy7wpz9myK15D4VA3t9QD7iwrVu3mnvuucdUrFjRBAUFmaJFi5qmTZuat956y+u2r2lpaWb48OGmUqVKJjAw0MTFxZkhQ4Z4rWPM2dvenvs7O8ZkfTvbrG49nHHL4l9//dXzOxSlSpUyQ4cO9bld6KRJk0zVqlVNcHCwqV69upk8eXK2tyzN7la6st0CPCUlxfzrX/8yiYmJpmjRoiY8PNwkJiZm+ZtGs2bNMvXq1TPBwcGmePHi5o477jB79uzxWsd++2W7rHLMztixY0316tVNYGCgKVWqlHnggQd8bvd6MbcAz27d7D6/d955x9SvX9+EhoaaokWLmtq1a5vBgwebvXv3etbZv3+/ad++vSlatKiR5HU78OPHj5shQ4aYKlWqmKCgIFOyZElz9dVXm1dffdXzW0tZ9Qe7JUuWmKZNm5rQ0FATERFhOnToYDZt2uSz3tKlS029evVMUFCQiY+PN++9954ZNGiQCQkJ8Vn3o48+Mtdcc40JDw834eHhpnr16qZ///5my5YtnnWaN29uatas6fPcrG6bm5qaakaNGmVq1qxpgoODTbFixUz9+vXN8OHDTXJystf7bL8FuDHGvPvuu6Zy5crG6XRmeTvw//znP0aSuffee7N8f7KSXR+0y+599+czM8a/cbh06VJzyy23mLJly5qgoCBTtmxZ06NHD59bjP/666+mdevWJjg42PP7Qhm/mXLuLcCz+kyM8e8z8DefrPzdubB58+ZZ3ir/XDrPbfqff/55Y8zZz+jRRx81ZcuWNYGB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bdmyRbt371aTJk3yK1UAAIACreK/F+TZa+0a2f7CK52jd+/emjp1qu677z5NnDjRa1n//v01fvx49erVS1OmTMmlLC/OlClTNHDgwEJxi/URI0bo6aef1siRI/Wvf/0rv9O5LPju/shjbrdbkydPVq9evbwqu8jISPXt21ePPfaYli1bpjVr1qhPnz5q0qQJd7YDAAAoxOLi4jRz5kyva1zOnDmjpKQklS9fPh8zK5zef/99DR48WO+//35+p+J1vVFhlu9F0pIlS7R7927dfffdPstef/113XTTTerSpYuaNWum0qVLe52SBwAAgMLnyiuvVFxcnNf3urlz56p8+fKqV6+e17put1sjRozwXH+VmJioOXPmeJa7XC717dvXs7xatWp64403vLbRu3dvdezYUa+++qrKlCmjEiVKqH///kpLS7voNuzevVu33HKLihQpooiICHXt2lUHDhzwWufTTz9Vw4YNFRISopIlS6pTp06eZR988IEaNGigokWLqnTp0rr99ts9v++TE19++aVOnz6t5557TseOHdO3337rtdztduvll19WlSpVFBwcrPLly+vFF1/0LN+zZ4969Oih4sWLKzw8XA0aNNCqVaskZb5vdgMHDlSLFi08j1u0aKEBAwZo4MCBKlmypNq2bStJGj16tGrXrq3w8HDFxcXpwQcf1IkTJ7y29c0336hFixYKCwtTsWLF1LZtW/3111+aNm2aSpQo4XNvgo4dO+quu+7K8Xt0MfK9SLr++utljNEVV1zhsywkJETjxo3Tn3/+qZMnT2ru3LnnvR4JAAAAhcPdd9+tyZMnex6///776tOnj896I0aM0LRp0zRx4kRt3LhRjz76qO688059+eWXks4WAbGxsZo9e7Y2bdqkZ599Vk8++aT+85//eG1n2bJl+vXXX7Vs2TJNnTpVU6ZMuehT+txut2655Rb9+eef+vLLL7V48WLt2LFD3bp186yzYMECderUSe3atdPatWu1dOlSNWrUyLM8LS1Nzz//vNavX6/58+dr165dnrs858SkSZPUo0cPBQYGqkePHpo0aZLX8iFDhmjkyJF65plntGnTJiUlJalUqVKSpBMnTqh58+b6448/9Mknn2j9+vUaPHiw1w1J/DF16lQFBQXpm2++8ZxC6XA49Oabb2rjxo2aOnWqvvjiCw0ePNjznHXr1qlVq1aqUaOGVq5cqa+//lodOnSQy+XSbbfdJpfLpU8++cSz/sGDB7VgwYIsD6xcCgXm7naXyrFjxxQZGXnBO1gAAAAUFhl376pUqZJCQkK8lhWGa5KOHj2qd999V3FxcdqyZYuks7dE//3339WvXz9FRUVpypQpSklJUfHixbVkyRKva9L79eunU6dOKSkpKcvXGDBggPbv3+854tS7d28tX75cv/76q+cGGl27dpXD4dDMmTOz3Mb5rklavHixbrzxRu3cuVNxcXGSpE2bNqlmzZr6/vvv1bBhQ1199dWqXLmypk+f7tf7snr1ajVs2FDHjx9XkSJFznvjhgzHjh1T6dKltXLlSiUmJmrdunW69tprtW/fPhUpUkTHjx9XdHS0xo4dq379+vk8/5133tHjjz+uXbt2qXjx4j7LMz6r+fPne2IDBw7UunXrtHz5cklnjyQdO3ZMP/7443nbN2fOHN1///06fPiwJOn222/X7t279fXXX2e5/oMPPqhdu3bpf//7n6SzR6bGjRun7du3X/CGE+cbH/7WBvl+JAkAAAD/PNHR0Wrfvr2mTJmiyZMnq3379ipZsqTXOtu3b9epU6fUpk0bFSlSxPNv2rRp+vXXXz3rjRs3TvXr11d0dLSKFCmid955R7t37/baVs2aNb3uMFimTJmLOr1Nkn755RfFxcV5CiRJqlGjhqKiovTLL79IyjxSkp01a9aoQ4cOKl++vIoWLarmzZtLkk/e5/Phhx8qPj5eiYmJkqS6deuqQoUKmjVrlifPlJSUbPNYt26d6tWrl2WBlBP169f3iS1ZskStWrVSuXLlVLRoUd111106cuSITp065Xnt870/99xzjxYtWqQ//vhD0tmitXfv3rlyRz5/FIi72wEAAOCf5+6779aAAQMknS10zpVxDcuCBQtUrlw5r2UZP/kyc+ZMPf7443rttdfUpEkTFS1aVK+88ornupoMgYGBXo8ty8rxaWU5ERoamu2ykydPqm3btmrbtq1mzJih6Oho7d69W23bts3RjQ8mTZqkjRs3et38zO126/3331ffvn3Pm8OFcpTOnjJ37klnWV3HlfFbRBl27dqlm266SQ888IBefPFFFS9eXF9//bX69u2r1NRUhYWFXfC169Wrp8TERE2bNk3XX3+9Nm7cqAUL8u4oKUUSAAAA8sUNN9yg1NRUWZblueDfrkaNGgoODtbu3bs9R1rO9c033+jqq6/Wgw8+6InZjzJdCgkJCfr999/1+++/e51ud/ToUdWoUUOSVKdOHS1dujTL66w2b96sI0eOaOTIkZ7nr169Okc5/Pzzz1q9erWWL1/udSTozz//VIsWLbR582ZVrVpVoaGhWrp0aZan29WpU0fvvfee/vzzzyyPJkVHR2vDhg1esXXr1vkUnOdas2aN3G63XnvtNc9viZ17jVjG+zN8+PBst9OvXz+NGTNGf/zxh1q3bu115O5So0gCACCH8vKaj8LsYq5XwT+L0+n0nJ6W1Y/tFi1aVI8//rgeffRRud1uXXPNNUpOTtY333yjiIgI9erVS1WrVtW0adP0+eefq1KlSvrggw/0ww8/qFKlSn87P5fLpXXr1nnFgoOD1bp1a9WuXVt33HGHxowZo/T0dD344INq3ry5GjRoIEkaOnSoWrVqpfj4eHXv3l3p6en63//+pyeeeELly5dXUFCQ3nrrLd1///3asGGDnn/++RzlNmnSJDVq1EjNmjXzWdawYUNNmjRJr7zyip544gkNHjxYQUFBatq0qQ4dOqSNGzeqb9++6tGjh1566SV17NhRI0aMUJkyZbR27VqVLVtWTZo0UcuWLfXKK69o2rRpatKkiaZPn64NGzb43IHwXFWqVFFaWpreeustdejQweuGDhmGDBmi2rVr68EHH9T999+voKAgLVu2TLfddpvntMvbb79djz/+uN59911NmzYtR+/P38U1SQAAAMg3ERER572A/vnnn9czzzyjESNGKCEhQTfccIMWLFjgKYLuu+8+de7cWd26ddNVV12lI0eOeB1V+jtOnDihevXqef3r0KGDLMvSxx9/rGLFiqlZs2Zq3bq1Kleu7LkWSDp7Q4PZs2frk08+Ud26ddWyZUt9//33ks4eoZkyZYpmz56tGjVqaOTIkXr11Vf9zis1NVXTp09Xly5dslzepUsXTZs2TWlpaXrmmWc0aNAgPfvss0pISFC3bt0812IFBQVp0aJFiomJUbt27VS7dm2NHDnSU7C2bdtWzzzzjAYPHuy5qUTPnj0vmF9iYqJGjx6tUaNGqVatWpoxY4ZGjBjhtc4VV1yhRYsWaf369WrUqJGaNGmijz/+2Od3U7t06aIiRYr43Ir8UuPudgAA5BBHkvzDkaRL53x37wIuJ61atVLNmjX15ptv+v2c3Li7HafbAQAAAChQ/vrrLy1fvlzLly/X+PHj8/z1KZIAAADg5ac9R/M7hUKjTmxUfqdwWapXr57++usvjRo1StWqVcvz16dIAgAAAFCg7Nq1K19fnyIpj3Eeu384jx0AAAD5hbvbAQAAAIANRRIAAAAA2FAkAQAAAIANRRIAAAAA2FAkAQAAAIANRRIAAAAKhL633aSXhw3J1W1OGD1SXdtem6vbxOWPW4ADAABcToZF/u1N1PFzvZ/6/ZbjbT/z6IP6ZM6HPvFPV6zR6Hc+UEBg3n89Xbrwv5o84Q3t3L5FbrdR6XKxanJtCw0eNiLPc8kN1atX186dO/Xbb7+pdOnS+Z1OoUSRBAAAgDzVtEUrPffaOK9YsRIl5XQ68zyXVV9/qcH979ZDg59WizbjJcvSjq1b9N1Xy/7WdtPS0hQYGJhLWWZKTU1VUFBQtsu//vprnT59WrfeequmTp2qJ554ItdzyIlL9T5capxuBwAAgDwVFBSskjGlvP45nU6f0+1ubFJH7731mp4dNEBNqsep7VW1NGfGFK9tvf7SUHVo1kBXVS2rdk3rauwrLyotLc3vXL5c8pnqNrhKve9/WBXjq6pi5SpqeUN7Pfniq17rLfv8f+p2Y3M1rFJa7ZrW1cTXRyk9Pd2z3LIsTZgwQTfffLPCw8P1/PPPKzY2VhMmTPDaztq1a+VwOPTbb2ePwh09elT9+vVTdHS0IiIi1LJlS61fv96z/rBhw1S3bl299957qlSpkkJCQs7bnkmTJun222/XXXfdpffff99n+Z49e9SjRw8VL15c4eHhatCggVatWuVZ/umnn6phw4YKCQlRyZIl1alTJ682zp8/32t7UVFRmjJliiRp165dsixLs2bNUvPmzRUSEqIZM2boyJEj6tGjh8qVK6ewsDDVrl1bH37ofTTR7Xbr5ZdfVpUqVRQcHKzy5cvrxRdflCS1bNlSAwYM8Fr/0KFDCgoK0tKlS8/7flwsiiQAAAAUWNPeGaeadepq1sIv1bVnX7345CDt+nWbZ3l4eFE9P3qc5n7xnQYPG6G5H07T9PfG+739EtEx+nXrZm3bvCnbdX5c9a2efvR+3XH3/Zq39Ds9M+J1fTw7Se+99ZrXesOGDVOnTp30888/q1+/furRo4eSkpK81pkxY4aaNm2qChUqSJJuu+02HTx4UAsXLtSaNWt05ZVXqlWrVvrzzz89z9m+fbs++ugjzZ07V+vWrcs2z+PHj2v27Nm688471aZNGyUnJ+urr77yLD9x4oSaN2+uP/74Q5988onWr1+vwYMHy+12S5IWLFigTp06qV27dlq7dq2WLl2qRo0a+f1eZvj3v/+tRx55RL/88ovatm2rM2fOqH79+lqwYIE2bNige++9V3fddZe+//57z3OGDBmikSNH6plnntGmTZuUlJSkUqVKSZL69eunpKQkpaSkeNafPn26ypUrp5YtW+Y4P39wuh0AAADy1Iqln6txtVjP42uua61XJ07Jct1rWrZRt179JEl3PzhQ09+boO+//UoV46tKku595HHPuuXiyuu3X7frs0/mqs8Dj/iVS48+92rt9yt1a5umKhsbp9r1GqhJs5Zq3+k2BQUHS5ImjnlZdz84UDff1kOSFFuhovo//qTGvDhM41/LvG7p9ttvV58+fTyP77jjDr322mvavXu3ypcvL7fbrZkzZ+rpp5+WdPbUuO+//14HDx5U8P+/1quvvqr58+drzpw5uvfeeyWdPcVu2rRpio6OPm9bZs6cqapVq6pmzZqSpO7du2vSpEm69tqzN65ISkrSoUOH9MMPP6h48eKSpCpVqnie/+KLL6p79+4aPny4J5aYmOjX+2g3cOBAde7c2Sv2+OOZn9NDDz2kzz//XP/5z3/UqFEjHT9+XG+88YbGjh2rXr16SZLi4+N1zTXXSJI6d+6sAQMG6OOPP1bXrl0lSVOmTFHv3r1lWVaO8/MHRRIAAADyVMOrr9VTL2YehQkNC8t23SsSanr+37IslYyO0Z9HDntin30yVx9Oflu//7ZLp06elMuVrvAiRf3OJSwsXGOn/ke/79qpH1Z+pZ9+XK3XXnhaSe9P1LSPFyk0NExbN23Quh9W6d23Rnue53a5lJJyRqdOnVLY/+ffoEEDr23XrVtXCQkJSkpK0r///W99+eWXOnjwoG677TZJ0vr163XixAmVKFHC63mnT5/Wr7/+6nlcoUKFCxZIkvT+++/rzjvv9Dy+88471bx5c7311lsqWrSo1q1bp3r16nkKpHOtW7dO99xzzwVf50LOfR9cLpdeeukl/ec//9Eff/yh1NRUpaSkeN63X375RSkpKWrVqlWW2wsJCfGcPti1a1f9+OOP2rBhgz755JO/nWt2KJIAAACQp0JDw1S+UmW/1g0I8L7o37Ismf8/PWz9mu/15MP36oHH/q2rm7dSkYgIffbxXH3w7tgc5xRXsZLiKlZS5x491e+hQbqleQN9/sk8dex2h06dPKkHBv1brW7o4PM8+zVC4eHhPsvvuOMOT5GUlJSkG264wVMUnThxQmXKlNHy5ct9nhcVFXXe7Z5r06ZN+u677/T999973azB5XJp5syZuueeexQaGnrebVxouWVZMsZ4xbK6/uvcfF955RW98cYbGjNmjGrXrq3w8HANHDhQqampfr2udPaUu7p162rPnj2aPHmyWrZs6Tll8VKgSAIAAEChtG719ypTLk73PJx5Kte+P37/29stF1deIaGhOn36pCQpoXYd7fp1e5aFncNx/kv8b7/9dj399NNas2aN5syZo4kTJ3qWXXnlldq/f78CAgJUsWLFv5XzpEmT1KxZM40b533XwMmTJ2vSpEm65557VKdOHb333nv6888/szyaVKdOHS1dutTrlEG76Oho7du3z/N427ZtOnXq1AVz++abb3TLLbd4jnK53W5t3bpVNWrUkCRVrVpVoaGhWrp0qfr165flNmrXrq0GDRro3XffVVJSksaOzXkhnBMUSQAAACiUKlSqrP1792jhxx+pVuKVWvHFIn3x2X9ztI0Jo0fqzOlTuqbl9SpTLk7HjyUr6f23lZ6WribXXidJuveRwXq4T3eVKRer1u1ulsPh0JZNG/Trll/0zpuvnnf7FStW1NVXX62+ffvK5XLp5ptv9ixr3bq1mjRpoo4dO+rll1/WFVdcob1793puoHDuaWvZSUtL0wcffKDnnntOtWrV8lrWr18/jR49Whs3blSPHj300ksvqWPHjhoxYoTKlCmjtWvXqmzZsmrSpImGDh2qVq1aKT4+Xt27d1d6err+97//eY5MtWzZUmPHjlWTJk3kcrn0xBNP+HV776pVq2rOnDn69ttvVaxYMY0ePVoHDhzwFEkhISF64oknNHjwYAUFBalp06Y6dOiQNm7cqL59+3q1ZcCAAQoPD/e6696lwN3tAAAAUCi1uL6d7uz3gEY+M1hdb2im9atX6d5H/pWjbdRv3FR7dv+mpwfer47XNVL/nrfpyKEDmjDjI8/NIZq2aKU3J8/UyhVf6I6bWumuW9po+nsTVCY2zq/XuOOOO7R+/Xp16tTJ69Qyy7L0v//9T82aNVOf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VKig0aNHKzQ0VH369PE6KiRJjRo10nXXXadRo0ZluY2sjiTFxcXpyJEjioiIkJS/ex8rD1lQqPeW5NUeoF+C+xTuvSV5tQfo6SPeuReAvY+SlPDsZ4W270l5N562h/bxiheqvpeX4+np/QVi76M9zlzOXH65z+UJz3529rUKad+TmMvPxgvQeCogc/mxY8dUokQJJScne2qDrOT76XZ2UVFRuuKKK7R9+3a1adNGqampOnr0qNfRpAMHDmR5DVOG4OBgBQcH+8QDAgIUEODd3Iw3+VwZb6a/8XO3e764kaU0t++62cXdsuTOKm4subMob13GkiuLeLqxzo5kP+Npbss3KGWTe3bxi29TgEn1xDMG/rnODmRf2cXt27xw3GQZt7KJO+SWI8v42cnMJ55bbcpB38subllWlvHsxoc/cXv/KWx9z+5Sj6dC3ffycjz9/9yb0z7JXJ7/44m53L94QZzLz+07ha3v2TGXF5DxVEDm8uyW++Tj11p55MSJE/r1119VpkwZ1a9fX4GBgVq6dKln+ZYtW7R79241adIkH7MEAAAAcDnL1yNJjz/+uDp06KAKFSpo7969Gjp0qJxOp3r06KHIyEj17dtXjz32mIoXL66IiAg99NBDatKkCXe2AwAAAHDJ5GuRtGfPHvXo0UNHjhxRdHS0rrnmGn333XeKjo6WJL3++utyOBzq0qWLUlJS1LZtW40fPz4/UwYAAABwmcvXImnmzJnnXR4SEqJx48Zp3LhxeZQRAAAAgH+6AnVNEgAAAADkN4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALAJyMnKbrdbX375pb766iv99ttvOnXqlKKjo1WvXj21bt1acXFxlypPAAAAAMgTfh1JOn36tF544QXFxcWpXbt2WrhwoY4ePSqn06nt27dr6NChqlSpktq1a6fvvvvuUucMAAAAAJeMX0eSrrjiCjVp0kTvvvuu2rRpo8DAQJ91fvvtNyUlJal79+566qmndM899+R6sgAAAABwqflVJC1atEgJCQnnXadChQoaMmSIHn/8ce3evTtXkgMAAACAvObX6XYXKpDsAgMDFR8ff9EJAQAAAEB+ytGNG+zS09P19ttva/ny5XK5XGratKn69++vkJCQ3MwPAAAAAPLURRdJDz/8sLZu3arOnTsrLS1N06ZN0+rVq/Xhhx/mZn4AAAAAkKf8LpLmzZunTp06eR4vWrRIW7ZskdPplCS1bdtWjRs3zv0MAQAAACAP+f1jsu+//746duyovXv3SpKuvPJK3X///frss8/06aefavDgwWrYsOElSxQAAAAA8oLfRdKnn36qHj16qEWLFnrrrbf0zjvvKCIiQk899ZSeeeYZxcXFKSkp6VLmCgAAAACXXI6uSerWrZvatm2rwYMHq23btpo4caJee+21S5UbAAAAAOQ5v48kZYiKitI777yjV155RT179tS//vUvnTlz5lLkBgAAAAB5zu8iaffu3eratatq166tO+64Q1WrVtWaNWsUFhamxMRELVy48FLmCQAAAAB5wu8iqWfPnnI4HHrllVcUExOj++67T0FBQRo+fLjmz5+vESNGqGvXrhedyMiRI2VZlgYOHOiJnTlzRv3791eJEiVUpEgRdenSRQcOHLjo1wAAAACAC/H7mqTVq1dr/fr1io+PV9u2bVWpUiXPsoSEBK1YsULvvPPORSXxww8/6O2331adOnW84o8++qgWLFig2bNnKzIyUgMGDFDnzp31zTffXNTrAAAAAMCF+H0kqX79+nr22We1aNEiPfHEE6pdu7bPOvfee2+OEzhx4oTuuOMOvfvuuypWrJgnnpycrEmTJmn06NFq2bKl6tevr8mTJ+vbb7/Vd999l+PXAQAAAAB/+H0kadq0aRo0aJAeffRR1a1bV2+//XauJNC/f3+1b99erVu31gsvvOCJr1mzRmlpaWrdurUnVr16dZUvX14rV67M9odrU1JSlJKS4nl87NgxSVJ6errS09MlSQ6HQw6HQ263W26327NuRtzlcskYc8G40+mU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",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def run_all_measures():\n",
    "    measures = [\"ratio\", \"kl\", \"js\", \"entropy_diff\"]\n",
    "    \n",
    "    results = {\n",
    "        \"measure\": [],\n",
    "        \"mean_local_acc\": [],\n",
    "        \"final_server_acc\": []\n",
    "    }\n",
    "    \n",
    "    for m in measures:\n",
    "        client_accs, server_acc, _ = run_fedchill_het(metric=m)\n",
    "        \n",
    "        \n",
    "        mean_local = np.mean(client_accs)\n",
    "        \n",
    "        results[\"measure\"].append(m)\n",
    "        results[\"mean_local_acc\"].append(mean_local)\n",
    "        results[\"final_server_acc\"].append(server_acc)\n",
    "    \n",
    "    return results\n",
    "\n",
    "print(\"Running FedChill with S = 0.5\")\n",
    "results3 = run_all_measures()\n",
    "plot_results(results3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T13:48:35.801174Z",
     "iopub.status.busy": "2025-09-16T13:48:35.800874Z",
     "iopub.status.idle": "2025-09-16T13:48:35.805597Z",
     "shell.execute_reply": "2025-09-16T13:48:35.804909Z",
     "shell.execute_reply.started": "2025-09-16T13:48:35.801153Z"
    },
    "trusted": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'measure': ['ratio', 'kl', 'js', 'entropy_diff'], 'mean_local_acc': [67.296875, 63.635416666666664, 68.25520833333334, 67.69270833333333], 'final_server_acc': [54.5, 54.26, 51.29, 50.73]}\n"
     ]
    }
   ],
   "source": [
    "print(results3)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Running FedChill (only Temp. init) with Different Measures"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Clients 10, Frac 0.1, Rounds 30, S=-3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T15:37:19.345756Z",
     "iopub.status.busy": "2025-09-16T15:37:19.344991Z",
     "iopub.status.idle": "2025-09-16T15:37:19.358545Z",
     "shell.execute_reply": "2025-09-16T15:37:19.357841Z",
     "shell.execute_reply.started": "2025-09-16T15:37:19.345714Z"
    },
    "trusted": true
   },
   "outputs": [],
   "source": [
    "def run_fedchill_het_ti(metric=\"ratio\"):\n",
    "    print_flag = False\n",
    "\n",
    "    client_train_loaders, client_val_loaders, test_loader, client_class_distributions = load_and_partition_data(\n",
    "        num_clients=NUM_OF_CLIENTS, \n",
    "        alpha=ALPHA, \n",
    "        batch_size=BATCH_SIZE, \n",
    "        frac=FRAC, \n",
    "        rand_seed=42\n",
    "    )\n",
    "    num_clients = len(client_train_loaders)\n",
    "    num_clients = len(client_train_loaders)\n",
    "\n",
    "    \n",
    "    global_distribution = calculate_global_distribution(client_class_distributions)\n",
    "    \n",
    "    \n",
    "    client_params = [AdaptiveClientHyperparams() for _ in range(num_clients)]\n",
    "    \n",
    "    for i, distribution in enumerate(client_class_distributions):\n",
    "        p = np.array([distribution.get(c, 0) for c in range(10)])\n",
    "        q = np.array([global_distribution.get(c, 1/10) for c in range(10)])\n",
    "\n",
    "        if metric == \"ratio\":\n",
    "            het_score = calculate_heterogeneity_score(distribution, global_distribution)\n",
    "        elif metric == \"kl\":\n",
    "            het_score = kl_divergence(p, q)\n",
    "        elif metric == \"js\":\n",
    "            het_score = js_divergence(p, q)\n",
    "        elif metric == \"l1\":\n",
    "            het_score = l1_distance(p, q)\n",
    "        elif metric == \"entropy_diff\":\n",
    "            het_score = entropy_diff(p, q)\n",
    "        else:\n",
    "            raise ValueError(f\"Unknown heterogeneity metric: {metric}\")\n",
    "\n",
    "        client_params[i].update_temp_from_heterogeneity(het_score)\n",
    "        print(f\"Client {i} - Het Score: {het_score:.4f}, Initial temp: {client_params[i].get_temp():.4f}\")\n",
    "    \n",
    "    \n",
    "    global_model = SimpleCNN().to(device)\n",
    "    round_server_acc_list = []\n",
    "    round_client_acc_list = [[] for _ in range(num_clients)]\n",
    "    \n",
    "    \n",
    "    client_temp_history = [[] for _ in range(num_clients)]\n",
    "    client_lr_history = [[] for _ in range(num_clients)]\n",
    "\n",
    "    client_models = [copy.deepcopy(global_model).to(device) for _ in range(num_clients)]\n",
    "    client_optimizers = [optim.SGD(model.parameters(), lr=client_params[i].get_lr()) for i, model in enumerate(client_models)]\n",
    "\n",
    "    for comm_round in range(COMM_ROUND):\n",
    "        print(f\"\\n{'='*20} COMMUNICATION ROUND {comm_round+1} {'='*20}\")\n",
    "\n",
    "        \n",
    "        selected_clients = list(range(num_clients))\n",
    "\n",
    "        \n",
    "        if comm_round == COMM_ROUND - 1:\n",
    "            print_flag = True\n",
    "            print(\"\\n--- LOCAL TRAINING OF CLIENTS ---\")\n",
    "\n",
    "        for client_idx in selected_clients:\n",
    "            \n",
    "            client_models[client_idx].load_state_dict(global_model.state_dict())\n",
    "            \n",
    "            \n",
    "            curr_lr = client_params[client_idx].get_lr()\n",
    "            curr_temp = client_params[client_idx].get_temp()\n",
    "            \n",
    "            \n",
    "            client_temp_history[client_idx].append(curr_temp)\n",
    "            client_lr_history[client_idx].append(curr_lr)\n",
    "            \n",
    "            \n",
    "            client_optimizers[client_idx] = optim.SGD(client_models[client_idx].parameters(), lr=curr_lr)\n",
    "\n",
    "            if comm_round == COMM_ROUND - 1:\n",
    "                print(f\"\\n--- Client {client_idx} Local Training with T={curr_temp:.4f}, LR={curr_lr:.6f} ---\")\n",
    "\n",
    "            \n",
    "            client_train_with_temp(\n",
    "                client_models[client_idx],\n",
    "                client_train_loaders[client_idx],\n",
    "                client_optimizers[client_idx],\n",
    "                LOCAL_EPOCHS,\n",
    "                temperature=curr_temp,\n",
    "                print_flag=print_flag\n",
    "            )\n",
    "\n",
    "            \n",
    "            local_acc = evaluate_model(client_models[client_idx], client_val_loaders[client_idx])\n",
    "            if comm_round == COMM_ROUND - 1:\n",
    "                print(f\"Client {client_idx} Private Data Validation Accuracy: {local_acc:.2f}%\")\n",
    "\n",
    "            \n",
    "            client_testset_acc = evaluate_model(client_models[client_idx], test_loader)\n",
    "            round_client_acc_list[client_idx].append(client_testset_acc)\n",
    "            \n",
    "            \n",
    "            \n",
    "            \n",
    "            \n",
    "\n",
    "        \n",
    "        aggregate_models(global_model, client_models, selected_clients, client_train_loaders)\n",
    "\n",
    "        \n",
    "        server_test_val_acc = evaluate_model(global_model, test_loader)\n",
    "        if comm_round == COMM_ROUND - 1:\n",
    "            print(f\"\\n--- Server Evaluation ---\")\n",
    "            print(f\"Server Test Data Validation Accuracy: {server_test_val_acc:.2f}%\")\n",
    "\n",
    "        round_server_acc_list.append(server_test_val_acc)\n",
    "\n",
    "\n",
    "    \n",
    "    client_final_accs = []\n",
    "    print(\"\\n--- Clients Final Evaluation on Private Data sets ---\")\n",
    "    for i in range(num_clients):\n",
    "        local_acc = evaluate_model(client_models[i], client_val_loaders[i])\n",
    "        client_final_accs.append(local_acc)\n",
    "        print(f\"Client {i} Final Private Data Validation Accuracy: {local_acc:.2f}%\")\n",
    "\n",
    "    final_test_acc = evaluate_model(global_model, test_loader)\n",
    "    print(f\"\\nFinal Server Test Accuracy: {final_test_acc:.2f}%\")\n",
    "\n",
    "    return client_final_accs, final_test_acc, global_model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T11:25:40.021138Z",
     "iopub.status.busy": "2025-09-16T11:25:40.020402Z",
     "iopub.status.idle": "2025-09-16T12:30:45.009223Z",
     "shell.execute_reply": "2025-09-16T12:30:45.008423Z",
     "shell.execute_reply.started": "2025-09-16T11:25:40.021115Z"
    },
    "trusted": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = ratio\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 0.9007, Initial temp: 0.0671\n",
      "Client 1 - Het Score: 0.7295, Initial temp: 0.1121\n",
      "Client 2 - Het Score: 0.8682, Initial temp: 0.0739\n",
      "Client 3 - Het Score: 0.8552, Initial temp: 0.0769\n",
      "Client 4 - Het Score: 0.5698, Initial temp: 0.1810\n",
      "Client 5 - Het Score: 0.5956, Initial temp: 0.1675\n",
      "Client 6 - Het Score: 0.7451, Initial temp: 0.1070\n",
      "Client 7 - Het Score: 0.6556, Initial temp: 0.1399\n",
      "Client 8 - Het Score: 1.0000, Initial temp: 0.0500\n",
      "Client 9 - Het Score: 1.0000, Initial temp: 0.0500\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.0671, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2463\n",
      "Epoch [2/5], Loss: 0.1233\n",
      "Epoch [3/5], Loss: 0.1727\n",
      "Epoch [4/5], Loss: 0.8099\n",
      "Epoch [5/5], Loss: 0.2856\n",
      "Client 0 Private Data Validation Accuracy: 59.06%\n",
      "\n",
      "--- Client 1 Local Training with T=0.1121, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.1155\n",
      "Epoch [2/5], Loss: 0.2939\n",
      "Epoch [3/5], Loss: 0.1763\n",
      "Epoch [4/5], Loss: 0.0903\n",
      "Epoch [5/5], Loss: 0.0940\n",
      "Client 1 Private Data Validation Accuracy: 75.00%\n",
      "\n",
      "--- Client 2 Local Training with T=0.0739, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.6021\n",
      "Epoch [2/5], Loss: 0.2191\n",
      "Epoch [3/5], Loss: 0.2055\n",
      "Epoch [4/5], Loss: 0.1311\n",
      "Epoch [5/5], Loss: 0.1700\n",
      "Client 2 Private Data Validation Accuracy: 67.71%\n",
      "\n",
      "--- Client 3 Local Training with T=0.0769, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3894\n",
      "Epoch [2/5], Loss: 0.0777\n",
      "Epoch [3/5], Loss: 0.0505\n",
      "Epoch [4/5], Loss: 0.0349\n",
      "Epoch [5/5], Loss: 0.0116\n",
      "Client 3 Private Data Validation Accuracy: 92.19%\n",
      "\n",
      "--- Client 4 Local Training with T=0.1810, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2678\n",
      "Epoch [2/5], Loss: 0.0992\n",
      "Epoch [3/5], Loss: 0.0572\n",
      "Epoch [4/5], Loss: 0.0333\n",
      "Epoch [5/5], Loss: 0.0296\n",
      "Client 4 Private Data Validation Accuracy: 63.54%\n",
      "\n",
      "--- Client 5 Local Training with T=0.1675, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.6295\n",
      "Epoch [2/5], Loss: 0.3580\n",
      "Epoch [3/5], Loss: 0.2217\n",
      "Epoch [4/5], Loss: 0.1249\n",
      "Epoch [5/5], Loss: 0.0647\n",
      "Client 5 Private Data Validation Accuracy: 60.16%\n",
      "\n",
      "--- Client 6 Local Training with T=0.1070, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.1657\n",
      "Epoch [2/5], Loss: 0.2867\n",
      "Epoch [3/5], Loss: 0.2400\n",
      "Epoch [4/5], Loss: 0.1090\n",
      "Epoch [5/5], Loss: 0.0813\n",
      "Client 6 Private Data Validation Accuracy: 65.62%\n",
      "\n",
      "--- Client 7 Local Training with T=0.1399, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3670\n",
      "Epoch [2/5], Loss: 0.1143\n",
      "Epoch [3/5], Loss: 0.0684\n",
      "Epoch [4/5], Loss: 0.0542\n",
      "Epoch [5/5], Loss: 0.0424\n",
      "Client 7 Private Data Validation Accuracy: 68.23%\n",
      "\n",
      "--- Client 8 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.4438\n",
      "Epoch [2/5], Loss: 0.0869\n",
      "Epoch [3/5], Loss: 0.1776\n",
      "Epoch [4/5], Loss: 0.1683\n",
      "Epoch [5/5], Loss: 0.0895\n",
      "Client 8 Private Data Validation Accuracy: 80.08%\n",
      "\n",
      "--- Client 9 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.5552\n",
      "Epoch [2/5], Loss: 0.5842\n",
      "Epoch [3/5], Loss: 0.5108\n",
      "Epoch [4/5], Loss: 0.3579\n",
      "Epoch [5/5], Loss: 0.1505\n",
      "Client 9 Private Data Validation Accuracy: 69.53%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 50.98%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 59.06%\n",
      "Client 1 Final Private Data Validation Accuracy: 73.44%\n",
      "Client 2 Final Private Data Validation Accuracy: 66.67%\n",
      "Client 3 Final Private Data Validation Accuracy: 91.67%\n",
      "Client 4 Final Private Data Validation Accuracy: 64.06%\n",
      "Client 5 Final Private Data Validation Accuracy: 60.16%\n",
      "Client 6 Final Private Data Validation Accuracy: 60.94%\n",
      "Client 7 Final Private Data Validation Accuracy: 69.27%\n",
      "Client 8 Final Private Data Validation Accuracy: 80.08%\n",
      "Client 9 Final Private Data Validation Accuracy: 72.66%\n",
      "\n",
      "Final Server Test Accuracy: 50.98%\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = kl\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 1.0227, Initial temp: 0.0500\n",
      "Client 1 - Het Score: 0.6109, Initial temp: 0.1600\n",
      "Client 2 - Het Score: 1.4072, Initial temp: 0.0500\n",
      "Client 3 - Het Score: 1.6845, Initial temp: 0.0500\n",
      "Client 4 - Het Score: 0.5174, Initial temp: 0.2118\n",
      "Client 5 - Het Score: 0.4435, Initial temp: 0.2643\n",
      "Client 6 - Het Score: 0.7640, Initial temp: 0.1011\n",
      "Client 7 - Het Score: 0.3035, Initial temp: 0.4023\n",
      "Client 8 - Het Score: 1.1237, Initial temp: 0.0500\n",
      "Client 9 - Het Score: 0.7487, Initial temp: 0.1058\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.0298\n",
      "Epoch [2/5], Loss: 0.6033\n",
      "Epoch [3/5], Loss: 0.6545\n",
      "Epoch [4/5], Loss: 0.4706\n",
      "Epoch [5/5], Loss: 0.5416\n",
      "Client 0 Private Data Validation Accuracy: 50.62%\n",
      "\n",
      "--- Client 1 Local Training with T=0.1600, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.6377\n",
      "Epoch [2/5], Loss: 0.7479\n",
      "Epoch [3/5], Loss: 0.5710\n",
      "Epoch [4/5], Loss: 0.3818\n",
      "Epoch [5/5], Loss: 0.3263\n",
      "Client 1 Private Data Validation Accuracy: 73.44%\n",
      "\n",
      "--- Client 2 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.4354\n",
      "Epoch [2/5], Loss: 0.6434\n",
      "Epoch [3/5], Loss: 0.6779\n",
      "Epoch [4/5], Loss: 0.3573\n",
      "Epoch [5/5], Loss: 0.3282\n",
      "Client 2 Private Data Validation Accuracy: 70.83%\n",
      "\n",
      "--- Client 3 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.4145\n",
      "Epoch [2/5], Loss: 0.2798\n",
      "Epoch [3/5], Loss: 0.1221\n",
      "Epoch [4/5], Loss: 0.1167\n",
      "Epoch [5/5], Loss: 0.0214\n",
      "Client 3 Private Data Validation Accuracy: 89.58%\n",
      "\n",
      "--- Client 4 Local Training with T=0.2118, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.8927\n",
      "Epoch [2/5], Loss: 0.3749\n",
      "Epoch [3/5], Loss: 0.2478\n",
      "Epoch [4/5], Loss: 0.1371\n",
      "Epoch [5/5], Loss: 0.1079\n",
      "Client 4 Private Data Validation Accuracy: 62.50%\n",
      "\n",
      "--- Client 5 Local Training with T=0.2643, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.4171\n",
      "Epoch [2/5], Loss: 0.9663\n",
      "Epoch [3/5], Loss: 0.6574\n",
      "Epoch [4/5], Loss: 0.4835\n",
      "Epoch [5/5], Loss: 0.3594\n",
      "Client 5 Private Data Validation Accuracy: 57.03%\n",
      "\n",
      "--- Client 6 Local Training with T=0.1011, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.7699\n",
      "Epoch [2/5], Loss: 0.7531\n",
      "Epoch [3/5], Loss: 0.4908\n",
      "Epoch [4/5], Loss: 0.3456\n",
      "Epoch [5/5], Loss: 0.2767\n",
      "Client 6 Private Data Validation Accuracy: 64.06%\n",
      "\n",
      "--- Client 7 Local Training with T=0.4023, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.4583\n",
      "Epoch [2/5], Loss: 0.7437\n",
      "Epoch [3/5], Loss: 0.4791\n",
      "Epoch [4/5], Loss: 0.3355\n",
      "Epoch [5/5], Loss: 0.2405\n",
      "Client 7 Private Data Validation Accuracy: 66.15%\n",
      "\n",
      "--- Client 8 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.8342\n",
      "Epoch [2/5], Loss: 0.2613\n",
      "Epoch [3/5], Loss: 0.2030\n",
      "Epoch [4/5], Loss: 0.1399\n",
      "Epoch [5/5], Loss: 0.1818\n",
      "Client 8 Private Data Validation Accuracy: 80.08%\n",
      "\n",
      "--- Client 9 Local Training with T=0.1058, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.9903\n",
      "Epoch [2/5], Loss: 0.4103\n",
      "Epoch [3/5], Loss: 0.2649\n",
      "Epoch [4/5], Loss: 0.1703\n",
      "Epoch [5/5], Loss: 0.1609\n",
      "Client 9 Private Data Validation Accuracy: 64.06%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 47.41%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 50.62%\n",
      "Client 1 Final Private Data Validation Accuracy: 76.56%\n",
      "Client 2 Final Private Data Validation Accuracy: 70.31%\n",
      "Client 3 Final Private Data Validation Accuracy: 90.10%\n",
      "Client 4 Final Private Data Validation Accuracy: 63.02%\n",
      "Client 5 Final Private Data Validation Accuracy: 59.38%\n",
      "Client 6 Final Private Data Validation Accuracy: 60.94%\n",
      "Client 7 Final Private Data Validation Accuracy: 64.58%\n",
      "Client 8 Final Private Data Validation Accuracy: 79.69%\n",
      "Client 9 Final Private Data Validation Accuracy: 66.41%\n",
      "\n",
      "Final Server Test Accuracy: 47.41%\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = js\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 0.2739, Initial temp: 0.4397\n",
      "Client 1 - Het Score: 0.1681, Initial temp: 0.6039\n",
      "Client 2 - Het Score: 0.3461, Initial temp: 0.3541\n",
      "Client 3 - Het Score: 0.4343, Initial temp: 0.2718\n",
      "Client 4 - Het Score: 0.1533, Initial temp: 0.6314\n",
      "Client 5 - Het Score: 0.1192, Initial temp: 0.6994\n",
      "Client 6 - Het Score: 0.1991, Initial temp: 0.5503\n",
      "Client 7 - Het Score: 0.0819, Initial temp: 0.7822\n",
      "Client 8 - Het Score: 0.2538, Initial temp: 0.4671\n",
      "Client 9 - Het Score: 0.1906, Initial temp: 0.5644\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.4397, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.0681\n",
      "Epoch [2/5], Loss: 0.0277\n",
      "Epoch [3/5], Loss: 0.0216\n",
      "Epoch [4/5], Loss: 0.0098\n",
      "Epoch [5/5], Loss: 0.0070\n",
      "Client 0 Private Data Validation Accuracy: 62.50%\n",
      "\n",
      "--- Client 1 Local Training with T=0.6039, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.6830\n",
      "Epoch [2/5], Loss: 0.2590\n",
      "Epoch [3/5], Loss: 0.1100\n",
      "Epoch [4/5], Loss: 0.0563\n",
      "Epoch [5/5], Loss: 0.0449\n",
      "Client 1 Private Data Validation Accuracy: 71.88%\n",
      "\n",
      "--- Client 2 Local Training with T=0.3541, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.4163\n",
      "Epoch [2/5], Loss: 0.1084\n",
      "Epoch [3/5], Loss: 0.0979\n",
      "Epoch [4/5], Loss: 0.0716\n",
      "Epoch [5/5], Loss: 0.0270\n",
      "Client 2 Private Data Validation Accuracy: 68.23%\n",
      "\n",
      "--- Client 3 Local Training with T=0.2718, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.4101\n",
      "Epoch [2/5], Loss: 0.0357\n",
      "Epoch [3/5], Loss: 0.0109\n",
      "Epoch [4/5], Loss: 0.0124\n",
      "Epoch [5/5], Loss: 0.0036\n",
      "Client 3 Private Data Validation Accuracy: 93.23%\n",
      "\n",
      "--- Client 4 Local Training with T=0.6314, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1178\n",
      "Epoch [2/5], Loss: 0.0450\n",
      "Epoch [3/5], Loss: 0.0261\n",
      "Epoch [4/5], Loss: 0.0196\n",
      "Epoch [5/5], Loss: 0.0168\n",
      "Client 4 Private Data Validation Accuracy: 64.58%\n",
      "\n",
      "--- Client 5 Local Training with T=0.6994, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3046\n",
      "Epoch [2/5], Loss: 0.1379\n",
      "Epoch [3/5], Loss: 0.0780\n",
      "Epoch [4/5], Loss: 0.0497\n",
      "Epoch [5/5], Loss: 0.0448\n",
      "Client 5 Private Data Validation Accuracy: 62.50%\n",
      "\n",
      "--- Client 6 Local Training with T=0.5503, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.8981\n",
      "Epoch [2/5], Loss: 0.2478\n",
      "Epoch [3/5], Loss: 0.0870\n",
      "Epoch [4/5], Loss: 0.0434\n",
      "Epoch [5/5], Loss: 0.0238\n",
      "Client 6 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Client 7 Local Training with T=0.7822, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1473\n",
      "Epoch [2/5], Loss: 0.0627\n",
      "Epoch [3/5], Loss: 0.0422\n",
      "Epoch [4/5], Loss: 0.0344\n",
      "Epoch [5/5], Loss: 0.0286\n",
      "Client 7 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Client 8 Local Training with T=0.4671, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1458\n",
      "Epoch [2/5], Loss: 0.0189\n",
      "Epoch [3/5], Loss: 0.0137\n",
      "Epoch [4/5], Loss: 0.0109\n",
      "Epoch [5/5], Loss: 0.0145\n",
      "Client 8 Private Data Validation Accuracy: 79.69%\n",
      "\n",
      "--- Client 9 Local Training with T=0.5644, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2418\n",
      "Epoch [2/5], Loss: 0.0474\n",
      "Epoch [3/5], Loss: 0.0325\n",
      "Epoch [4/5], Loss: 0.0207\n",
      "Epoch [5/5], Loss: 0.0140\n",
      "Client 9 Private Data Validation Accuracy: 69.53%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 55.97%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 62.50%\n",
      "Client 1 Final Private Data Validation Accuracy: 70.31%\n",
      "Client 2 Final Private Data Validation Accuracy: 68.23%\n",
      "Client 3 Final Private Data Validation Accuracy: 92.71%\n",
      "Client 4 Final Private Data Validation Accuracy: 65.62%\n",
      "Client 5 Final Private Data Validation Accuracy: 57.03%\n",
      "Client 6 Final Private Data Validation Accuracy: 65.62%\n",
      "Client 7 Final Private Data Validation Accuracy: 69.27%\n",
      "Client 8 Final Private Data Validation Accuracy: 79.30%\n",
      "Client 9 Final Private Data Validation Accuracy: 72.66%\n",
      "\n",
      "Final Server Test Accuracy: 55.97%\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = entropy_diff\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 0.8527, Initial temp: 0.0774\n",
      "Client 1 - Het Score: 0.7037, Initial temp: 0.1211\n",
      "Client 2 - Het Score: 1.3864, Initial temp: 0.0500\n",
      "Client 3 - Het Score: 1.6167, Initial temp: 0.0500\n",
      "Client 4 - Het Score: 0.5006, Initial temp: 0.2227\n",
      "Client 5 - Het Score: 0.4251, Initial temp: 0.2793\n",
      "Client 6 - Het Score: 0.9316, Initial temp: 0.0611\n",
      "Client 7 - Het Score: 0.3201, Initial temp: 0.3827\n",
      "Client 8 - Het Score: 1.0807, Initial temp: 0.0500\n",
      "Client 9 - Het Score: 0.8083, Initial temp: 0.0885\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.0774, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2209\n",
      "Epoch [2/5], Loss: 0.0982\n",
      "Epoch [3/5], Loss: 0.0884\n",
      "Epoch [4/5], Loss: 0.1255\n",
      "Epoch [5/5], Loss: 0.1181\n",
      "Client 0 Private Data Validation Accuracy: 57.19%\n",
      "\n",
      "--- Client 1 Local Training with T=0.1211, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.0916\n",
      "Epoch [2/5], Loss: 0.3458\n",
      "Epoch [3/5], Loss: 0.1764\n",
      "Epoch [4/5], Loss: 0.1228\n",
      "Epoch [5/5], Loss: 0.1534\n",
      "Client 1 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Client 2 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.1493\n",
      "Epoch [2/5], Loss: 0.3119\n",
      "Epoch [3/5], Loss: 0.3124\n",
      "Epoch [4/5], Loss: 0.2127\n",
      "Epoch [5/5], Loss: 0.1430\n",
      "Client 2 Private Data Validation Accuracy: 68.23%\n",
      "\n",
      "--- Client 3 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.5735\n",
      "Epoch [2/5], Loss: 0.1028\n",
      "Epoch [3/5], Loss: 0.0685\n",
      "Epoch [4/5], Loss: 0.0440\n",
      "Epoch [5/5], Loss: 0.0278\n",
      "Client 3 Private Data Validation Accuracy: 93.75%\n",
      "\n",
      "--- Client 4 Local Training with T=0.2227, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2902\n",
      "Epoch [2/5], Loss: 0.0743\n",
      "Epoch [3/5], Loss: 0.0674\n",
      "Epoch [4/5], Loss: 0.0217\n",
      "Epoch [5/5], Loss: 0.0215\n",
      "Client 4 Private Data Validation Accuracy: 63.54%\n",
      "\n",
      "--- Client 5 Local Training with T=0.2793, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.7478\n",
      "Epoch [2/5], Loss: 0.2988\n",
      "Epoch [3/5], Loss: 0.1568\n",
      "Epoch [4/5], Loss: 0.0957\n",
      "Epoch [5/5], Loss: 0.0811\n",
      "Client 5 Private Data Validation Accuracy: 63.28%\n",
      "\n",
      "--- Client 6 Local Training with T=0.0611, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.4839\n",
      "Epoch [2/5], Loss: 0.3163\n",
      "Epoch [3/5], Loss: 0.2508\n",
      "Epoch [4/5], Loss: 0.1938\n",
      "Epoch [5/5], Loss: 0.2115\n",
      "Client 6 Private Data Validation Accuracy: 59.38%\n",
      "\n",
      "--- Client 7 Local Training with T=0.3827, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.4660\n",
      "Epoch [2/5], Loss: 0.1378\n",
      "Epoch [3/5], Loss: 0.1001\n",
      "Epoch [4/5], Loss: 0.0801\n",
      "Epoch [5/5], Loss: 0.0704\n",
      "Client 7 Private Data Validation Accuracy: 66.15%\n",
      "\n",
      "--- Client 8 Local Training with T=0.0500, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.5721\n",
      "Epoch [2/5], Loss: 0.1841\n",
      "Epoch [3/5], Loss: 0.1322\n",
      "Epoch [4/5], Loss: 0.0574\n",
      "Epoch [5/5], Loss: 0.0938\n",
      "Client 8 Private Data Validation Accuracy: 63.67%\n",
      "\n",
      "--- Client 9 Local Training with T=0.0885, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.5160\n",
      "Epoch [2/5], Loss: 0.2156\n",
      "Epoch [3/5], Loss: 0.5067\n",
      "Epoch [4/5], Loss: 0.1252\n",
      "Epoch [5/5], Loss: 0.0713\n",
      "Client 9 Private Data Validation Accuracy: 64.06%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 49.94%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 57.19%\n",
      "Client 1 Final Private Data Validation Accuracy: 70.31%\n",
      "Client 2 Final Private Data Validation Accuracy: 67.71%\n",
      "Client 3 Final Private Data Validation Accuracy: 93.75%\n",
      "Client 4 Final Private Data Validation Accuracy: 63.54%\n",
      "Client 5 Final Private Data Validation Accuracy: 58.59%\n",
      "Client 6 Final Private Data Validation Accuracy: 57.81%\n",
      "Client 7 Final Private Data Validation Accuracy: 66.15%\n",
      "Client 8 Final Private Data Validation Accuracy: 64.06%\n",
      "Client 9 Final Private Data Validation Accuracy: 65.62%\n",
      "\n",
      "Final Server Test Accuracy: 49.94%\n"
     ]
    },
    {
     "data": {
      "image/png": "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nCY4c+ZMde3aVTfccINiY2M1c+ZMv+t9++23at++vUqUKKHo6GjVqlVLL7/8stc6mzdvVrdu3RQfH6/IyEhVr15djz76qGd5bteZ+Zs3z9aeQOcC6ez9Mb/mPRQhBihELrnkElOlSpWA1+/Tp4+RZLp27WomTJhgevfubSSZjh07eq2XlJRkqlevbsqXL29GjBhhXnrpJXPJJZeYYsWKmbfffttUrFjRjBkzxowZM8bExsaaqlWrGpfL5fU6ERERplq1aua2224z48ePNzfccIORZB5//HGv10pISDD33HOPGT9+vHnxxRdNo0aNjCTz3//+12s9SSY1NdXEx8ebkSNHmgkTJpg1a9Z4lg0fPtyz7i233GLCwsLMgw8+aP7973+bsWPHmg4dOpi3337bs87UqVONJNOwYUPz0ksvmX/9618mMjLSVKpUyfz5558+bbnssstM//79zcSJE02XLl2MJPPaa6+d8z0fPny4kWTatGljXn31VTNo0CDjdDpNw4YNTUZGhjHGmHnz5plOnToZSWbixInmrbfeMuvWrTvnNrds2WIOHjzo8y8xMdFcf/31Xs8ZMGCACQkJMXfccYeZNGmSefjhh010dLRPHgkJCSYlJcW89dZb5q233jKLFi0yxhhz/PhxU6tWLVOqVCnzyCOPmEmTJpnevXsby7LM/fff73mdHTt2GEmmRo0apkqVKmbMmDHmpZdeMr/88otZvHixCQkJMZdeeql59tlnzciRI03p0qVNiRIlzI4dOzzb+P777014eLipVKmSGTNmjHn66adNhQoVTO3atc2Z0/BTTz1lLMsy3bt3N6+99ppnm2d+js2bNzcVKlQwiYmJ5v777zevvfaaadWqlZFk/ve//3nWc7lc5pprrjFRUVFm8ODB5vXXXzeDBg0yISEh5qabbvJ67aSkJNOnTx9jjDE///yzue+++4wk88gjj3jev3379pnFixcbSeajjz7yev7evXuN0+k0o0aNyvWzNuZ0H4yOjjZHjhwxERERZsqUKZ5lgwcPNikpKZ73/bnnnvMsC/QzMyawcbh+/XoTFhZmGjRoYF5++WUzadIkM2TIENOsWTPPOtnjyv55GmPMZ599ZiSZzz77zOszKVeunImPjzf33nuvef311838+fMD/gwCyeds72le5sKyZcuaRx55xIwfP97Uq1fPWJZl1q9ff87XkWQGDhyY63K3221atWplLMsyAwYMMOPHjzcdOnQwkszgwYO91h0xYoSRZK688krz3HPPmZdfftnccsst5uGHH/as07FjR9OtWzfz3HPPmYkTJ5qbb77ZSDJDhgzxaX9SUtI58zfGmGHDhhnLsrzGSV64XC4TGRlp6tevbw4dOnTOdQMdf7n9Tejfv7+Ji4sz6enpXutPnz7dSDIrV67Mt9c6m2+++cZIMl988YUxxpj+/fubGjVq+Ky3aNEiExYWZpKSkszw4cPNxIkTzX333WfatGnjWWfdunUmJibGlCpVygwbNsy8/vrrZujQoaZmzZqedXL7TLP/XgTankD/Jp+rP/7VeQ9FD0USCo20tDQjyWcyz83atWuNJDNgwACv+JAhQ4wk8+mnn3piSUlJRpL5+uuvPbFPPvnESDKRkZHml19+8cRff/11ny8/2V9A7r33Xk/M7Xab66+/3oSFhZmDBw964idOnPDKJyMjw1x++eWmVatWXnFJxuFwmA0bNvi07cwiKTY29qxfTDIyMkyZMmXM5Zdfbk6ePOmJ//e//zWSzBNPPOHTljMn9Lp165r69evn+hrGGHPgwAETFhZmrrnmGq8icvz48UaSefPNNz2x7D9k9vcmN9nrnu2fvUj64osvjCTzzjvveG3n448/9olfdtllpnnz5j6v+eSTT5ro6GizdetWr/i//vUv43Q6za5du4wxOUVSTEyMOXDggNe6derUMWXKlPH6orRu3TrjcDhM7969PbEOHTqYqKgo89tvv3li27ZtMyEhIV5/7Hfu3GmcTqd5+umnvV7nxx9/NCEhIV7x5s2bG0lmxowZnlh6eropV66c6dKliyf21ltvGYfD4flik23SpElGkvnqq688MXuRZIwxs2fP9hkLxpz+MpaQkGC6d+/uFX/xxReNZVlm+/bt5myyiyRjjOnatatp3bq1Z7vlypUzI0eO9FskBfqZGRPYOHzppZfO2UfzWiRJMpMmTfJaN9DPIJB8/DmfuXD58uWe2IEDB0x4eLh56KGHzvla5yqS5s+fbySZp556yivetWtXY1mW+emnn4wxp/u/w+EwnTp18ppLjDk9t2Y783M0xpi77rrLREVFmVOnTnligRZJmZmZ5tZbbzVhYWEmOjra629CXjzxxBNGkomOjjbXXXedefrpp83q1at91svL+Mvtb0L236ozv5y3b9/ea6difrzW2QwaNMgkJiZ6Pp9FixYZSV7FVVZWlqlcubJJSkry2qljjPfn2qxZM1O8eHGvv71nrpPXIim39gQyFwTSH//qvIeih9PtUGgcOXJEklS8ePGA1v/f//4nSXrwwQe94g899JAk+Vy7VKNGDTVp0sTz+IorrpAktWrVShUrVvSJb9++3ec17Xd1yj68n5GRoSVLlnjikZGRnv//+eefSktL09VXX+1zapwkNW/eXDVq1DhHS09f1/Ptt99qz549fpevWrVKBw4c0D333ON1Dvb111+vlJQUv9dx/eMf//B6fPXVV/tts92SJUuUkZGhwYMHe13gf8cddygmJuYvXy/2/vvva/HixT7/ypYt67Xe7NmzFRsbq7Zt2+r333/3/Ktfv76KFSumzz777JyvNXv2bF199dUqUaKE1zbatGkjl8ul5cuXe63fpUsXxcfHex7v3btXa9euVd++fVWyZElPvFatWmrbtq2nf7pcLi1ZskQdO3ZUhQoVPOtVrVpV1113nddrzJ07V263W926dfPKqVy5cqpWrZpPu4oVK6Zbb73V8zgsLEyNGjXy+hxnz56t1NRUpaSkeG2zVatWkhTQe3Umh8OhXr166cMPP9TRo0c98XfeeUdXXnmlKleuHPC2brnlFi1btkz79u3Tp59+qn379uV6ql1ePrNAxmH29XIffPCBz+l65ys8PNzn1KhAP4Pzzed85sKrr77a8zg+Pl7Vq1c/5/gPNBen06n77rvPJxdjjBYuXChJmj9/vtxut5544gmvuUSS16lU9s/x6NGj+v3333X11VfrxIkT2rx5c57zGzp0qBYuXKgff/xRV1xxhdq3b6+1a9d6lu/du1eWZXldr+TPyJEjNXPmTNWtW1effPKJHn30UdWvX1/16tXzOt02r+PP39+EVq1aqXTp0po1a5Yn9ueff2rx4sXq3r17vr5WbrKysjRr1ix1797d8/m0atVKZcqU0TvvvONZb82aNdqxY4cGDx7scz1q9vMOHjyo5cuXq3///l5/e+3rnI/c2hPIXBBIf8zPeQ9FAzduQKERExMjSV6Tz9n88ssvcjgcPndOK1eunOLi4vTLL794xc+cjGNjYyVJiYmJfuP2a1uk0xNklSpVvGKXXnqpJHldr/Df//5XTz31lNauXet1Hri/yT/QSfXZZ59Vnz59lJiYqPr166t9+/bq3bu3J5/stvo7JzolJUVffvmlVywiIsLrC78klShRwqfNZ8rtdcLCwlSlShWf9zyvmjVrptKlS/vEz7z4dtu2bUpLS/M59z9bIBdOb9u2TT/88IPP+5DbNs78rM72nqempuqTTz7R8ePHdeTIEZ08edLvHf7OjG3btk3GGFWrVs1vTqGhoV6PExISfPpViRIl9MMPP3htc9OmTQG3M1C9e/fW2LFjNW/ePPXu3VtbtmzR6tWrNWnSpDxtp3379ipevLhmzZqltWvXqmHDhqpatarf247n5TMLZBx2795d//73vzVgwAD961//UuvWrdW5c2d17drV54tSoC655BKfG3oE+hmcbz5/dS6UAhv/gfjll19UoUIFn51dqampnuXS6Z95cDgc5/ySvmHDBj322GP69NNPPTvSsqWlpeUpt99++02vvPKKRo8erUsvvVTz589X8+bNdc011+iLL75Q9erVtX79ekk5O8vOpmfPnurZs6eOHDmib7/9VtOmTdPMmTPVoUMHrV+/XhEREXkef/7+JoSEhKhLly6aOXOm0tPTFR4errlz5yozM9OrSMqP18rNokWLdPDgQTVq1Eg//fSTJ96yZUu9++67Gjt2rBwOh37++WdJ0uWXX57rtrKL8bOtcz5ya08gc0Gg/TG/5j0UDRRJKDRiYmJUoUIFzx+pQAW658npdOYpbvxcZHwuX3zxhW688UY1a9ZMr732msqXL6/Q0FBNnTrV7wWu9j1cZ9OtWzddffXVmjdvnhYtWqTnnntOY8eO1dy5c32ORgQitzYXFW6322cPpl1uXxLO3Ebbtm01dOhQv8uzC+BsgX5Wf4Xb7ZZlWVq4cKHfz6hYsWJejwPpu263WzVr1tSLL77od90zdxIEqkaNGqpfv77efvtt9e7dW2+//bbCwsJ87lR3LuHh4ercubOmT5+u7du3n/UGH4F+ZoGOw8jISC1fvlyfffaZFixYoI8//lizZs1Sq1attGjRIjmdzlznl9xuHOCvnwT6GQSSz9n81bnwfOa8C+nw4cNq3ry5YmJiNGrUKCUnJysiIkLff/+9Hn744Twf/fv222/lcrnUuHFjSafPWli4cKGaNm2qNm3a6IsvvtAbb7yh2rVr5+kLfExMjNq2bau2bdsqNDRU06dP17fffqvmzZvnefzlNs/06NFDr7/+uhYuXKiOHTvqP//5j1JSUlS7dm3POvn1Wv5kz7W5je/PP/9cLVu2DHh7gciPsZfXv8nnkl/zHooGiiQUKjfccIPeeOMNrVixwuvUOH+SkpLkdru1bds2z15KSdq/f78OHz6spKSkfM3N7XZr+/btXl+et27dKkmeO/C8//77ioiI0CeffOJ1t6SpU6f+5dcvX7687rnnHt1zzz06cOCA6tWrp6efflrXXXedp61btmzxnFqRbcuWLfn2Xthfx35ULSMjQzt27FCbNm3y5XXOJTk5WUuWLFHTpk3P+Yc+tz+0ycnJOnbs2HnnbH8vzrR582aVLl1a0dHRnlu62/e+ZjszlpycLGOMKleu7FOkna/k5GStW7dOrVu3zvOpLOdav3fv3nrwwQe1d+9ez227S5Qokeccb7nlFr355ptyOBzq0aNHrusF+pnlZRw6HA61bt1arVu31osvvqhnnnlGjz76qD777DO1adPG0x77Hfgk5emoaV4+g3Pl409Bz4Vnk5SUpCVLlujo0aNeR5OyT43LziU5OVlut1sbN25UnTp1/G5r2bJlOnTokObOnatmzZp54jt27Div3LLfe/udIsuWLatPPvlETZs2VfPmzbV7927NnTv3vLYvSQ0aNND06dO1d+9eSX9t/Nk1a9ZM5cuX16xZs3TVVVfp008/9boTXH6+1pmOHz+uDz74QN27d1fXrl19lt93331655131LJlSyUnJ0uS1q9fn2t/zf7bca4doiVKlPAZd1Lexl6gc0Eg/TFbfs17KPy4JgmFytChQxUdHa0BAwZo//79Pst//vlnz21E27dvL0kaN26c1zrZe9Guv/76fM9v/Pjxnv8bYzR+/HiFhoaqdevWkuTZ82zf07Vz507Nnz//vF/T5XL5nFZSpkwZVahQwXPqQIMGDVSmTBlNmjTJ63SChQsXatOmTfn2XrRp00ZhYWF65ZVXvPY6T5kyRWlpaRfkPfenW7ducrlcevLJJ32WZWVlef1hjY6O9vuHtlu3blqxYoU++eQTn2WHDx9WVlbWWXMoX7686tSpo+nTp3ttf/369Vq0aJGnfzqdTrVp00bz58/3uqbsp59+8lyfka1z585yOp0aOXKkz159Y4wOHTp01pz86datm3777TdNnjzZZ9nJkyd1/PjxXJ8bHR0tybdAyNazZ09ZlqX7779f27dv97o+Ki9atmypJ598UuPHj1e5cuVyXS/QzyzQcfjHH3/4bCf7C1L2OMr+0me/3snlcuXpB60D/QwCycefYMyFZ8vF5XJ5zZWS9NJLL8myLM+R744dO8rhcGjUqFE+R4Sy+372ES/7WMjIyNBrr712XrldddVVCg8P15gxY3TixAlPPDk5WePGjdOuXbsUGxur5s2bn3U7J06c0IoVK/wuyx7T2afh/pXxZ+dwONS1a1d99NFHeuutt5SVleV1ql1+vtaZ5s2bp+PHj2vgwIHq2rWrz78bbrhB77//vtLT01WvXj1VrlxZ48aN85k3sj/H+Ph4NWvWTG+++aZ27drldx3p9OeSlpbmdfrw3r17NW/evIBzD3QuCKQ/ZsuveQ+FH0eSUKgkJydr5syZ6t69u1JTU9W7d29dfvnlysjI0Ndff63Zs2d7fseldu3a6tOnj9544w3PaRnfffedpk+fro4dO+b7of+IiAh9/PHH6tOnj6644gotXLhQCxYs0COPPOI5vev666/Xiy++qGuvvVa33HKLDhw4oAkTJqhq1apeE31eHD16VAkJCeratatq166tYsWKacmSJVq5cqVeeOEFSaevVRk7dqz69eun5s2bq2fPntq/f79efvllVapUSQ888EC+vAfx8fEaNmyYRo4cqWuvvVY33nijtmzZotdee00NGzYssD8WzZs311133aXRo0dr7dq1uuaaaxQaGqpt27Zp9uzZevnllz17POvXr6+JEyfqqaeeUtWqVVWmTBm1atVK//znP/Xhhx/qhhtuUN++fVW/fn0dP35cP/74o+bMmaOdO3f6vT7K7rnnntN1112nJk2a6Pbbb9fJkyf16quvKjY21uu0sREjRmjRokVq2rSp7r77bs+XyMsvv9zrovHk5GQ99dRTGjZsmHbu3KmOHTuqePHi2rFjh+bNm6c777zT6zdRAnHbbbfpP//5j/7xj3/os88+U9OmTeVyubR582b95z//0SeffKIGDRr4fW6dOnXkdDo1duxYpaWlKTw83HOxtnS6P1x77bWaPXu24uLizvvLuMPh0GOPPXbO9QL9zAIdh6NGjdLy5ct1/fXXKykpSQcOHNBrr72mhIQEXXXVVZKkyy67TI0bN9awYcP0xx9/qGTJknrvvffOWUTbBfoZBJKPPwU9F65atUpPPfWUT7xFixbq0KGDWrZsqUcf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TpzQzJkz/b7GoEGDtG/fPs8Rp759+2rZsmX6+eefPT8u2q1bNzkcDr333nt+t3G2a5IWL16s6667Tjt27FBiYqIkaePGjbrsssv03XffqWHDhrryyitVpUoVvf322wG9L6tWrVLDhg119OhRFStWTMuWLVPLli31559/Ki4uzu9zjhw5onLlymnFihWqXbu21q5dq6uvvlp79+5VsWLFdPToUcXHx2v8+PEaMGCAz/PfeOMNDRkyRDt37lTJkiV9lmd/VvPnz/fEBg8erLVr12rZsmWSTh9JOnLkiL7//vuztm/OnDn6xz/+od9//12SdMstt2jXrl368ssv/a5/zz33aOfOnfrf//4n6fSRqQkTJuinn3465y2+zzY+Aq0NOJIEAACAAhcfH6/rr79e06ZN09SpU3X99derdOnSXuv89NNPOnHihNq2batixYp5/s2YMUM///yzZ70JEyaofv36io+PV7FixfTGG29o165dXtu67LLLPAWSJJUvX/68Tm+TpE2bNikxMdFTIElSjRo1FBcXp02bNknKOVKSm9WrV6tDhw6qWLGiihcvrubNm0uST95n8+677yo5OVm1a9eWJNWpU0dJSUmaNWuWJ8/09PRc81i7dq3q1q3rt0DKi/r16/vElixZotatW+uSSy5R8eLFddttt+nQoUM6ceKE57XP9v7ccccdWrRokX777TdJp4vWvn37FthvgwW1SKpUqZIsy/L5N3DgQEmnq8CBAweqVKlSKlasmLp06eJzricAAACKpv79+2vatGmaPn2639Oosq9hWbBggdauXev5t3HjRs9Rovfee09DhgzR7bffrkWLFmnt2rXq16+fzw0EQkNDvR5blpXn08ryIjIyMtdlx48fV7t27RQTE6N33nlHK1eu1Lx58yTl7cYHU6ZM0YYNGxQSEuL5t3HjRs8NHM6WQyDLHQ6HzjzpzN91XNHR0V6Pd+7cqRtuuEG1atXS+++/r9WrV2vChAmSctp3rteuW7euateurRkzZmj16tXasGHDeZ2OeL6CWiStXLlSe/fu9fxbvHixJOnmm2+WJD3wwAP66KOPNHv2bH3++efas2ePOnfuHMyUAQAAkE+uvfZaZWRkKDMz03PBv12NGjUUHh6uXbt2qWrVql7/so/ifPXVV7ryyit1zz33qG7duqpatarXUaYLITU1Vb/++qt+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tDp16uTVxvnz53ttLy4uTtOmTZMk7dy5U5ZladasWWrevLkiIiL0zjvv6NChQ+rZs6cuueQSRUVFqWbNmnr33Xe9tuN2u/Xss8+qatWqCg8PV8WKFfX0009Lklq1aqVBgwZ5rX/w4EGFhYVp6dKlZ30/zhdFEgAAAAqtF154QQ0aNNCaNWt0zz336O6779aWLVs8y4sXL65p06Zp48aNevnllzV58mS99NJLAW+/XLly2rBhg9avX5/rOl988YV69+6t+++/Xxs3btTrr7+uadOmeb7EZxsxYoQ6deqkH3/8UQMGDFDPnj01c+ZMr3XeeecdNW3aVElJSZKkm2++WQcOHNDChQu1evVq1atXT61bt9Yff/zhec5PP/2k999/X3PnztXatWtzzfPo0aOaPXu2br31VrVt21ZpaWn64osvPMuPHTum5s2b67ffftOHH36odevWaejQoXK73ZKkBQsWqFOnTmrfvr3WrFmjpUuXqlGjRgG/l9n+9a9/6f7779emTZvUrl07nTp1SvXr19eCBQu0fv163Xnnnbrtttv03XffeZ4zbNgwjRkzRo8//rg2btyomTNnqmzZspKkAQMGaObMmUpPT/es//bbb+uSSy5Rq1at8pxfIDjdDgAAAAXqv//9r4oVK+Z5fN1112n27Nl+123fvr3uueceSaePGr300kv67LPPVL16dUnSY4895lm3UqVKGjJkiN577z0NHTo0oFzuvfdeffHFF6pZs6aSkpLUuHFjXXPNNerVq5fCw8MlSSNHjtS//vUv9enTR5JUpUoVPfnkkxo6dKiGDx/u2dYtt9yifv36eR736tVLL7zwgnbt2qWKFSvK7Xbrvffe8+T85Zdf6rvvvtOBAwc8r/X8889r/vz5mjNnju68805Jp0+xmzFjhuLj48/alvfee0/VqlXTZZddJknq0aOHpkyZoquvvlqSNHPmTB08eFArV65UyZIlJUlVq1b1PP/pp59Wjx49NHLkSE+sdu3aAb2PdoMHD1bnzp29YkOGDPH8/95779Unn3yi//znP2rUqJGOHj2ql19+WePHj/e8x8nJybrqqqskSZ07d9agQYP0wQcfqFu3bpKkadOmqW/fvrIsK8/5BYIjSQAAAChQLVu21Nq1az3/XnnllVzXrVWrluf/lmWpXLlyOnDggCc2a9YsNW3aVOXKlVOxYsX02GOPadeuXQHnEh0drQULFuinn37SY489pmLFiumhhx5So0aNdOLECUnSunXrNGrUKBUrVszz74477tDevXs960hSgwYNvLZdp04dpaameo4mff755zpw4IBuvvlmz3aPHTumUqVKeW17x44d+vnnnz3bSUpKOmeBJElvvvmmbr31Vs/jW2+9VbNnz9bRo0clSWvXrlXdunU9BdKZ1q5dq9atWwfytp3Vme+Dy+XSk08+qZo1a6pkyZIqVqyYPvnkE8/ntGnTJqWnp+f62hEREV6nD37//fdav369+vbt+5dzzQ1HkgAAAFCgoqOjvY5gnM2ZF/1bluU5PWzFihXq1auXRo4cqXbt2ik2NlbvvfeeXnjhhTznlJycrOTkZA0YMECPPvqoLr30Us2aNUv9+vXTsWPHNHLkSJ+jI5K8rhGKjo72Wd6rVy/NnDlT//rXvzRz5kxde+21KlWqlKTTp7+VL19ey5Yt83leXFzcWbd7po0bN+qbb77Rd99953WzBpfLpffee0933HGHIiMjz7qNcy23LEvGGK+Yv+u/zsz3ueee08svv6xx48apZs2aio6O1uDBg5WRkRHQ60qnT7mrU6eOdu/eralTp6pVq1aeUxYvBIokAAAAFElff/21kpKS9Oijj3pi2TdE+CsqVaqkqKgoHT9+XJJUr149bdmyJeDCzu6WW27RY489ptWrV2vOnDmaNGmSZ1m9evW0b98+hYSEqFKlSn8p5ylTpqhZs2Y+N5yYOnWqpkyZojvuuEO1atXSv//9b/3xxx9+jybVqlVLS5cu9Tpl0C4+Pl579+71PN62bZvXkbTcfPXVV7rppps8R7ncbre2bt2qGjVqSJKqVaumyMhILV26VAMGDPC7jZo1a6pBgwaaPHmyZs6cqfHjx5/zdf8KiiQAAAAUSdWqVdOuXbv03nvvqWHDhlqwYIHmzZuXp22MGDFCJ06cUPv27ZWUlKTDhw/rlVdeUWZmptq2bStJeuKJJ3TDDTeoYsWK6tq1qxwOh9atW6f169frqaeeOuv2K1WqpCuvvFK33367XC6XbrzxRs+yNm3aqEmTJurYsaOeffZZXXrppdqzZ4/nBgpnnraWm8zMTL311lsaNWqULr/8cq9lAwYM0IsvvqgNGzaoZ8+eeuaZZ9SxY0eNHj1a5cuX15o1a1ShQgU1adJEw4cPV+vWrZWcnKwePXooKytL//vf/zxHplq1aqXx48erSZMmcrlcevjhhwO6vXe1atU0Z84cff311ypRooRefPFF7d+/31MkRURE6OGHH9bQoUMVFhampk2b6uDBg9qwYYNuv/12r7YMGjRI0dHRXnfduxC4JgkAAABF0o033qgHHnhAgwYNUp06dfT111/r8ccfz9M2mjdvru3bt6t3795KSUnRddddp3379mnRokWem0O0a9dO//3vf7Vo0SI1bNhQjRs31ksvvRTw6V69evXSunXr1KlTJ69TyyzL0v/+9z81a9ZM/fr106WXXqoePXrol19+8dzZLRAffvihDh065LdwSE1NVWpqqqZMmaKwsDAtWrRIZcqUUfv27VWzZk2NGTPG8/tULVq00OzZs/Xhhx+qTp06atWqldcd6F544QUlJibq6quv1i233KIhQ4YoKirqnPk99thjqlevntq1a6cWLVqoXLly6tixo9c6jz/+uB566CE98cQTSk1NVffu3b2uPZOknj17KiQkRD179jznrdD/KsuceWLhRebIkSOKjY1VWlqaYmJigp0OP7IZIH5ks2ijnweGfl500ccDQx+/cE6dOqUdO3YE9Ls5wMVi586dSk5O1sqVK1WvXr1c1zvb+Ai0NuB0OwAAAHj5YffhYKdQZNRKiAt2Che9zMxMHTp0SI899pgaN2581gIpv3C6HQAAAIBC66uvvlL58uW1cuVKrxtfXEgcSQIAAABQaLVo0cLn1uMXGkeSAAAAAMCGIgkAAKCIusjvvwWcl/wYFxRJAAAARUz2b9ME8kOewN9N9rgI5DeccsM1SQAAAEWM0+lUXFyc53dkoqKiZFlWvm3fZGXk27YudqdOnQp2Cvh/xhidOHFCBw4cUFxcnOf3n84HRRIAAEARVK5cOUny+cHN/HDgz5P5vs2LVdjJyHOvhAIVFxfnGR/niyIJAACgCLIsS+XLl1eZMmWUmZmZr9seMHdZvm7vYrb0oRbBTgE2oaGhf+kIUjaKJAAAgCLM6XTmy5dCu9+OuvJ1exeziIiIYKeAC4AbNwAAAACADUUSAAAAANhQJAEAAACADUUSAAAAANhQJAEAAACADUUSAAAAANhQJAEAAACADUUSAAAAANhQJAEAAACADUUSAAAAANhQJAEAAACADUUSAAAAANhQJAEAAACADUUSAAAAANhQJAEAAACA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kgQAAAAANhRJAAAAAGBDkQQAAAAANhRJAAAAAGBDkQQAAAAANhRJAAAAAGBDkQQAAAAANhRJAAAAAGBDkQQAAAAANhRJAAAAAGBDkQQAAAAANhRJAAAAAGBDkQQAAAAANhRJAAAAAGBDkQQAAAAANhRJAAAAAGBDkQQAAAAANhRJAAAAAGBDkQQAAAAANhRJAAAAAGATkpeV3W63Pv/8c33xxRf65ZdfdOLECcXHx6tu3bpq06aNEhMTL1SeAAAAAFAgAjqSdPLkST311FNKTExU+/bttXDhQh0+fFhOp1M//fSThg8frsqVK6t9+/b65ptvLnTOAAAAAHDBBHQk6dJLL1WTJk00efJktW3bVqGhoT7r/PLLL5o5c6Z69OihRx99VHfccUe+JwsAAAAAF1pARdKiRYuUmpp61nWSkpI0bNgwDRkyRLt27cqX5AAAAACgoAV0ut25CiS70NBQJScnn3dCAAAAABBMebpxg11WVpZef/11LVu2TC6XS02bNtXAgQMVERGRn/kBAAAAQIE67yLpvvvu09atW9W5c2dlZmZqxowZWrVqld599938zA8AAAAAClTARdK8efPUqVMnz+NFixZpy5YtcjqdkqR27dqpcePG+Z8hAAAAABSggH9M9s0331THjh21Z88eSVK9evX0j3/8Qx9//LE++ugjDR06VA0bNrxgiQIAAABAQQi4SProo4/Us2dPtWjRQq+++qreeOMNxcTE6NFHH9Xjjz+uxMREzZw580LmCgAAAAAXXJ6uSerevbvatWunoUOHql27dpo0aZJeeOGFC5UbAAAAABS4gI8kZYuLi9Mbb7yh5557Tr1799Y///lPnTp16kLkBgAAAAAFLuAiadeuXerWrZtq1qypXr16qVq1alq9erWioqJUu3ZtLVy48ELmCQAAAAAFIuAiqXfv3nI4HHruuedUpkwZ3XXXXQoLC9PIkSM1f/58jR49Wt26dbuQuQIAAADABRfwNUmrVq3SunXrlJycrHbt2qly5cqeZampqVq+fLneeOONC5IkAAAAABSUgIuk+vXr64knnlCfPn20ZMkS1axZ02edO++8M1+TAwAAAICCFvDpdjNmzFB6eroeeOAB/fbbb3r99dcvZF4AAAAAEBQBH0lKSkrSnDlzLmQuAAAAABB0AR1JOn78eJ42mtf1AQAAAKCwCKhIqlq1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yXgqJHN5bst98glorQvEGKNBgwZp3rx5+vTTT1W5cmWv5fXr11doaKiWLl3qiW3ZskW7du1SkyZNCjpdAAAAAH8DQT2SNHDgQM2cOVMffPCBihcv7rnOKDY2VpGRkYqNjdXtt9+uBx98UCVLllRMTIzuvfdeNWnShDvbAQAAALggglokTZw4UZLUokULr/jUqVPVt29fSdJLL70kh8OhLl26KD09Xe3atdNrr71WwJkCAAAA+LsIapEUyD0jIiIiNGHCBE2YMKEAMgIAAADwdxfUa5IAAAAAoLChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALAJapG0fPlydejQQRUqVJBlWZo/f77XcmOMnnjiCZUvX16RkZFq06aNtm3bFpxkAQAAAPwtBLVIOn78uGrXrq0JEyb4Xf7ss8/qlVde0aRJk/Ttt98qOjpa7dq106lTpwo4UwAAAAB/FyHBfPHrrrtO1113nd9lxhiNGzdOjz32mG666SZJ0owZM1S2bFnNnz9fPXr0KMhUAQAAAPxNFNprknbs2KF9+/apTZs2nlhsbKyuuOIKrVixIoiZAQAAALiYBfVI0tns27dPklS2bFmveNmyZT3L/ElPT1d6errn8ZEjRyRJWVlZysrKkiQ5HA45HA653W653W7Putlxl8slY8w5406nU5ZlebZrj0uSy+XyiVsyCjmjNM10Wz5xY6QsY8khI6e/uGXktHLibiO5jCWn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1axZU9HR0UpMTNQ999yjY8eOSZKWLVumfv36KS0tTZZlybIsjRgxQpLvoeZdu3bppptuUrFixRQTE6Nu3bpp//79QWglcH4WLFig2NhYvfPOO8FOBcg39tPt5syZo5o1ayoyMlKlSpVSmzZtdPz48eAmiCLH7XZr9OjRqly5siIjI1W7dm3NmTNH0unvDZZlaenSpWrQoIGioqJ05ZVXasuWLZJOn50ycuRIrVu3zvO9Ytq0aZJOn7I2ceJE3XjjjYqOjtbTTz8tSZo4caKSk5MVFham6tWr66233vLKJ/t51113nSIjI1WlShVPPpLUqlUrDRo0yOs5Bw8eVFhYmJYuXXrO9h44cEAdOnRQZGSkKleu7PdvhP10O8uytHr1ao0aNcrzvclfDHlHkYQLavr06QoLC9NXX32lSZMmyeFw6JVXXtGGDRs0ffp0ffrppxo6dKgk6corr9S4ceMUExOjvXv3au/evRoyZIjPNt1ut2666Sb98ccf+vzzz7V48WJt375d3bt3L+jmAedl5syZ6tmzp9555x316tUr2OkA+W7v3r3q2bOn+vfvr02bNmnZsmXq3Lkzp+Ihz0aPHq0ZM2Zo0qRJ2rBhgx544AHdeuut+vzzzz3rPProo3rhhRe0atUqhYSEqH///pKk7t2766GHHtJll13m+V5h/64wYsQIderUST/++KP69++vefPm6f7779dDDz2k9evX66677lK/fv302WefeeX0+OOPq0uXLlq3bp169eqlHj16aNOmTZKkAQMGaObMmUpPT/es//bbb+uSSy5Rq1atztnevn376tdff9Vnn32mOXPm6LXXXtOBAwdyXX/v3r267LLL9NBDD3m+N/mL4TwY4AJp3ry5qVu37lnXmT17tilVqpTn8dSpU01sbKzPeklJSeall14yxhizaNEi43Q6za5duzzLN2zYYCSZ7777Ll9yB/Jb8+bNzf3332/Gjx9vYmNjzbJlyzzL+vTpY2666abgJQfkk+y+vHr1aiPJ7Ny5M9gpoQg7deqUiYqKMl9//bVX/Pbbbzc9e/Y0n332mZFklixZ4lm2YMECI8mcPHnSGGPM8OHDTe3atX22LckMHjzYK3bllVeaO+64wyt28803m/bt23s97x//+IfXOldccYW5++67jTHGnDx50pQoUcLMmjXLs7xWrVpmxIgR52zvli1bfL7LbNq0yUjyfAfKzmHevHmex7Vr1zbDhw/32pa/GPKGI0m4oOrXr+/1eMmSJWrdurUuueQSFS9eXLfddpsOHTqkEydOBLzNTZs2KTExUYmJiZ5YjRo1FBcX59mTAxRGc+bM0QMPPKDFixerefPmwU4HuGBq166t1q1bq2bNmrr55ps1efJk/fnnn8FOC0XMTz/9pBMnTqht27YqVqyY59+MGTP0888/e9arVauW5//ly5eXpLMefcnWoEEDr8ebNm1S06ZNvWJNmzb1+W7RpEkTn8fZ60REROi2227Tm2++KUn6/vvvtX79evXt2/ec+WzatEkhISFe351SUlIUFxd3zuci/1Ek4YKy35Fu586duuGGG1SrVi29//77Wr16tSZMmCDp9E0dgItd3bp1FR8frzfffJPTjnBRczqdWrx4sRYuXKgaNWro1VdfVfXq1bVjx45gp4YiJPua5QULFmjt2rWefxs3bvS6Dsh+wwXLsiSdPjX/XC7UXXMHDBigxYsXa/fu3Zo6dapatWrFDRSKIIokFJjVq1fL7XbrhRdeUOPGjXXppZdqz549XuuEhYXJ5XKddTupqan69ddf9euvv3piGzdu1OHDh1WjRo0LkjuQH5KTk/XZZ5/pgw8+0L333hvsdIALyrIsNW3aVCNHjtSaNWsUFhamefPmBTstFCE1atRQeHi4du3apapVq3r9s59NcjaBfK/Ilpqaqq+++sor9tVXX/l8t/jmm298Hqempnoe16xZUw0aNNDkyZM1c+ZMzzVS55KSkqKsrCytXr3aE9uyZYvPz6KgYHALcBSYqlWrKjMzU6+++qo6dOjguZmDXaVKlXTs2DEtXbpUtWvXVlRUlKKiorzWadOmjWrWrKlevXpp3LhxysrK0j333KPmzZv7HDoHCptLL71Un332mVq0aKGQkBB+JBkXpW+//VZLly7VNddcozJlyujbb7/VwYMHvb5IAudSvHhxDRkyRA888IDcbreuuuoqpaWl6auvvlJMTExAR2cqVaqkHTt2aO3atUpISFDx4sUVHh7ud91//vOf6tatm+rWras2bdroo48+0ty5c7VkyRKv9WbPnq0GDRroqquu0jvvvKPvvvtOU6ZM8VpnwIABGjRokKKjo9WpU6eA2lu9enVde+21uuuuuzRx4kSFhIRo8ODBioyMDOj5yF8cSUKBqV27tl588UWNHTtWl19+ud555x2NHj3aa50rr7xS//jHP9S9e3fFx8fr2Wef9dmOZVn64IMPVKJECTVr1kxt2rRRlSpV9H/t3SGrIlEAxfEjz2CyKWgTRZEBgxhsalOTaJovIHbrGMRg0WA0CSKMJpPVoIjBKmaL4DcwKOy2Zd4+ZAV9zj74/2DS3AvnxsO9d2Y2m71rKcBTEomElsulbNtWs9l0Ow7wcn6/X6vVSuVyWfF4XJZlqd/vq1QquR0NP0yn01Gr1VK321UymVSxWNRisVAkEnlofq1WU7FYVKFQUCAQkG3bd8dWKhUNBgP1ej0ZhqHhcKjRaKR8Pv9pXLvd1nQ6VSqV0ng8lm3bX3abTNOU1+uVaZry+XwPr3c0GikcDiuXy6laraperysYDD48H6/j+cXBeAAA8AKmaerj40OTycTtKMC38Hg8ms/nf/4Hds/xeFQ0GtVut1M6nX5POLwUO0kAAOApt9tNh8NB2+1WhmG4HQdwzfV61fl8lmVZymazFKQfjJIEAACest/vlclkZBiGGo2G23EA12w2G4VCIe12uy/3rtfr9adPmf/94P/CcTsAAADgm10uF51Op7vvY7HYG9PgXyhJAAAAAODAcTsAAAAAcKAkAQAAAIADJQkAAAAAHChJAAAAAOBASQIAAAAAB0oSAAAAADhQkgAAAADAgZIEAAAAAA6/AcLUvpxD7/enAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "BATCH_SIZE = 64\n",
    "LEARNING_RATE = 0.01\n",
    "LOCAL_EPOCHS = 5\n",
    "NUM_OF_CLIENTS = 10\n",
    "COMM_ROUND = 30\n",
    "ALPHA = 0.5  \n",
    "FRAC = 0.1   \n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "def run_all_measures():\n",
    "    measures = [\"ratio\", \"kl\", \"js\", \"entropy_diff\"]\n",
    "    \n",
    "    results = {\n",
    "        \"measure\": [],\n",
    "        \"mean_local_acc\": [],\n",
    "        \"final_server_acc\": []\n",
    "    }\n",
    "    \n",
    "    for m in measures:\n",
    "        client_accs, server_acc, _ = run_fedchill_het_ti(metric=m)\n",
    "        \n",
    "        \n",
    "        mean_local = np.mean(client_accs)\n",
    "        \n",
    "        results[\"measure\"].append(m)\n",
    "        results[\"mean_local_acc\"].append(mean_local)\n",
    "        results[\"final_server_acc\"].append(server_acc)\n",
    "    \n",
    "    return results\n",
    "\n",
    "\n",
    "def plot_results(results):\n",
    "    measures = results[\"measure\"]\n",
    "    x = np.arange(len(measures))\n",
    "    \n",
    "    plt.figure(figsize=(10,6))\n",
    "    \n",
    "    \n",
    "    plt.bar(x - 0.2, results[\"mean_local_acc\"], width=0.4, label=\"Mean Local Accuracy\")\n",
    "    \n",
    "    \n",
    "    plt.bar(x + 0.2, results[\"final_server_acc\"], width=0.4, label=\"Final Server Accuracy\")\n",
    "    \n",
    "    plt.xticks(x, measures)\n",
    "    plt.ylabel(\"Accuracy (%)\")\n",
    "    plt.title(\"Comparison of Heterogeneity Measures on Local & Server Accuracy\")\n",
    "    plt.legend()\n",
    "    plt.grid(axis=\"y\", linestyle=\"--\", alpha=0.7)\n",
    "    plt.show()\n",
    "\n",
    "\n",
    "\n",
    "results2 = run_all_measures()\n",
    "plot_results(results2)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T12:31:04.199262Z",
     "iopub.status.busy": "2025-09-16T12:31:04.198929Z",
     "iopub.status.idle": "2025-09-16T12:31:04.205429Z",
     "shell.execute_reply": "2025-09-16T12:31:04.204840Z",
     "shell.execute_reply.started": "2025-09-16T12:31:04.199233Z"
    },
    "trusted": true
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'measure': ['ratio', 'kl', 'js', 'entropy_diff'],\n",
       " 'mean_local_acc': [69.79947916666667,\n",
       "  68.16145833333334,\n",
       "  70.32552083333333,\n",
       "  66.47395833333333],\n",
       " 'final_server_acc': [50.98, 47.41, 55.97, 49.94]}"
      ]
     },
     "execution_count": 37,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "results2\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Clients 10, Frac 0.1, Rounds 30, S=-0.5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T16:11:59.154977Z",
     "iopub.status.busy": "2025-09-16T16:11:59.154273Z",
     "iopub.status.idle": "2025-09-16T16:11:59.167440Z",
     "shell.execute_reply": "2025-09-16T16:11:59.166738Z",
     "shell.execute_reply.started": "2025-09-16T16:11:59.154952Z"
    },
    "trusted": true
   },
   "outputs": [],
   "source": [
    "class AdaptiveClientHyperparams:\n",
    "    def __init__(self, initial_lr=0.01, min_lr=0.0001, max_lr=0.05, \n",
    "                 initial_temp=0.5, min_temp=0.05, max_temp=1.0,\n",
    "                 patience=2, lr_decay_factor=0.7, temp_adjust_factor=0.8):\n",
    "        \n",
    "        self.lr = initial_lr\n",
    "        self.min_lr = min_lr\n",
    "        self.max_lr = max_lr\n",
    "        self.lr_decay_factor = lr_decay_factor\n",
    "        \n",
    "        \n",
    "        self.temp = initial_temp\n",
    "        self.min_temp = min_temp\n",
    "        self.max_temp = max_temp\n",
    "        self.temp_adjust_factor = temp_adjust_factor\n",
    "        \n",
    "        \n",
    "        self.patience = patience\n",
    "        self.performance_history = []\n",
    "        self.stagnation_count = 0\n",
    "        self.improvement_count = 0\n",
    "        self.het_score = 0.0    \n",
    "        \n",
    "    def update_temp_from_heterogeneity(self, het_score):\n",
    "        \"\"\"Initialize temperature based on client's heterogeneity score\"\"\"\n",
    "        self.het_score = het_score\n",
    "        \n",
    "        \n",
    "        self.temp = self.max_temp * math.exp(-0.5 * het_score) \n",
    "        self.temp = min(max(self.temp, self.min_temp), self.max_temp)\n",
    "        \n",
    "    def record_performance(self, accuracy):\n",
    "        \"\"\"Record performance and adjust hyperparameters if needed\"\"\"\n",
    "        self.performance_history.append(accuracy)\n",
    "        \n",
    "        \n",
    "        if len(self.performance_history) >= 3:\n",
    "            \n",
    "            if self.performance_history[-1] <= self.performance_history[-2]:\n",
    "                self.stagnation_count += 1\n",
    "                self.improvement_count = 0\n",
    "            else:\n",
    "                self.improvement_count += 1\n",
    "                self.stagnation_count = 0\n",
    "                \n",
    "            \n",
    "            if self.stagnation_count >= self.patience:\n",
    "                \n",
    "                if self.temp > self.min_temp * 1.1:  \n",
    "                    self.temp *= self.temp_adjust_factor\n",
    "                    self.temp = max(self.temp, self.min_temp)\n",
    "                    self.stagnation_count = 0\n",
    "                    return True, \"decrease-temp\", self.temp\n",
    "        \n",
    "        return False, \"unchanged\", None\n",
    "        \n",
    "    def get_lr(self):\n",
    "        return self.lr\n",
    "        \n",
    "    def get_temp(self):\n",
    "        return self.temp\n",
    "    \n",
    "\n",
    "def calculate_heterogeneity_score(client_distribution, global_distribution):\n",
    "    \"\"\"Calculate how different client's class distribution is from global distribution\"\"\"\n",
    "    score = 0\n",
    "    for cls in range(10):\n",
    "        client_prob = client_distribution.get(cls, 0)\n",
    "        global_prob = global_distribution.get(cls, 1/10)\n",
    "        if client_prob > 0 and global_prob > 0:\n",
    "            ratio = client_prob / global_prob\n",
    "            score += abs(ratio - 1)\n",
    "    \n",
    "    \n",
    "    score = min(score / 10, 1)\n",
    "    return score\n",
    "\n",
    "def calculate_global_distribution(client_class_distributions):\n",
    "    \"\"\"Calculate approximate global class distribution from client distributions\"\"\"\n",
    "    global_dist = {}\n",
    "    n_clients = len(client_class_distributions)\n",
    "    \n",
    "    for cls in range(10):\n",
    "        global_dist[cls] = sum(dist.get(cls, 0) for dist in client_class_distributions) / n_clients\n",
    "    \n",
    "    for cls in range(10):\n",
    "        if global_dist[cls] < 0.01:\n",
    "            global_dist[cls] = 0.01\n",
    "    \n",
    "    total = sum(global_dist.values())\n",
    "    global_dist = {k: v/total for k, v in global_dist.items()}\n",
    "    \n",
    "    return global_dist"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T17:10:08.070351Z",
     "iopub.status.busy": "2025-09-16T17:10:08.070099Z",
     "iopub.status.idle": "2025-09-16T18:14:45.157705Z",
     "shell.execute_reply": "2025-09-16T18:14:45.156855Z",
     "shell.execute_reply.started": "2025-09-16T17:10:08.070327Z"
    },
    "trusted": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = ratio\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 0.9007, Initial temp: 0.6374\n",
      "Client 1 - Het Score: 0.7295, Initial temp: 0.6944\n",
      "Client 2 - Het Score: 0.8682, Initial temp: 0.6478\n",
      "Client 3 - Het Score: 0.8552, Initial temp: 0.6521\n",
      "Client 4 - Het Score: 0.5698, Initial temp: 0.7521\n",
      "Client 5 - Het Score: 0.5956, Initial temp: 0.7424\n",
      "Client 6 - Het Score: 0.7451, Initial temp: 0.6890\n",
      "Client 7 - Het Score: 0.6556, Initial temp: 0.7205\n",
      "Client 8 - Het Score: 1.0000, Initial temp: 0.6065\n",
      "Client 9 - Het Score: 1.0000, Initial temp: 0.6065\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.6374, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.0651\n",
      "Epoch [2/5], Loss: 0.0308\n",
      "Epoch [3/5], Loss: 0.0167\n",
      "Epoch [4/5], Loss: 0.0122\n",
      "Epoch [5/5], Loss: 0.0099\n",
      "Client 0 Private Data Validation Accuracy: 57.81%\n",
      "\n",
      "--- Client 1 Local Training with T=0.6944, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.6371\n",
      "Epoch [2/5], Loss: 0.2551\n",
      "Epoch [3/5], Loss: 0.1472\n",
      "Epoch [4/5], Loss: 0.0673\n",
      "Epoch [5/5], Loss: 0.0483\n",
      "Client 1 Private Data Validation Accuracy: 65.62%\n",
      "\n",
      "--- Client 2 Local Training with T=0.6478, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3939\n",
      "Epoch [2/5], Loss: 0.0952\n",
      "Epoch [3/5], Loss: 0.0421\n",
      "Epoch [4/5], Loss: 0.0381\n",
      "Epoch [5/5], Loss: 0.0202\n",
      "Client 2 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Client 3 Local Training with T=0.6521, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2737\n",
      "Epoch [2/5], Loss: 0.0497\n",
      "Epoch [3/5], Loss: 0.0240\n",
      "Epoch [4/5], Loss: 0.0174\n",
      "Epoch [5/5], Loss: 0.0117\n",
      "Client 3 Private Data Validation Accuracy: 90.62%\n",
      "\n",
      "--- Client 4 Local Training with T=0.7521, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1079\n",
      "Epoch [2/5], Loss: 0.0456\n",
      "Epoch [3/5], Loss: 0.0296\n",
      "Epoch [4/5], Loss: 0.0205\n",
      "Epoch [5/5], Loss: 0.0197\n",
      "Client 4 Private Data Validation Accuracy: 61.98%\n",
      "\n",
      "--- Client 5 Local Training with T=0.7424, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2670\n",
      "Epoch [2/5], Loss: 0.1626\n",
      "Epoch [3/5], Loss: 0.0807\n",
      "Epoch [4/5], Loss: 0.0480\n",
      "Epoch [5/5], Loss: 0.0375\n",
      "Client 5 Private Data Validation Accuracy: 57.03%\n",
      "\n",
      "--- Client 6 Local Training with T=0.6890, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 1.1067\n",
      "Epoch [2/5], Loss: 0.3668\n",
      "Epoch [3/5], Loss: 0.1245\n",
      "Epoch [4/5], Loss: 0.0615\n",
      "Epoch [5/5], Loss: 0.0355\n",
      "Client 6 Private Data Validation Accuracy: 64.06%\n",
      "\n",
      "--- Client 7 Local Training with T=0.7205, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1193\n",
      "Epoch [2/5], Loss: 0.0410\n",
      "Epoch [3/5], Loss: 0.0255\n",
      "Epoch [4/5], Loss: 0.0220\n",
      "Epoch [5/5], Loss: 0.0194\n",
      "Client 7 Private Data Validation Accuracy: 72.92%\n",
      "\n",
      "--- Client 8 Local Training with T=0.6065, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1551\n",
      "Epoch [2/5], Loss: 0.0305\n",
      "Epoch [3/5], Loss: 0.0168\n",
      "Epoch [4/5], Loss: 0.0149\n",
      "Epoch [5/5], Loss: 0.0224\n",
      "Client 8 Private Data Validation Accuracy: 80.47%\n",
      "\n",
      "--- Client 9 Local Training with T=0.6065, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1828\n",
      "Epoch [2/5], Loss: 0.0488\n",
      "Epoch [3/5], Loss: 0.0267\n",
      "Epoch [4/5], Loss: 0.0188\n",
      "Epoch [5/5], Loss: 0.0124\n",
      "Client 9 Private Data Validation Accuracy: 67.97%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 54.89%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 57.81%\n",
      "Client 1 Final Private Data Validation Accuracy: 70.31%\n",
      "Client 2 Final Private Data Validation Accuracy: 69.79%\n",
      "Client 3 Final Private Data Validation Accuracy: 90.10%\n",
      "Client 4 Final Private Data Validation Accuracy: 61.98%\n",
      "Client 5 Final Private Data Validation Accuracy: 53.91%\n",
      "Client 6 Final Private Data Validation Accuracy: 64.06%\n",
      "Client 7 Final Private Data Validation Accuracy: 70.83%\n",
      "Client 8 Final Private Data Validation Accuracy: 80.08%\n",
      "Client 9 Final Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "Final Server Test Accuracy: 54.89%\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = kl\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 1.0227, Initial temp: 0.5997\n",
      "Client 1 - Het Score: 0.6109, Initial temp: 0.7368\n",
      "Client 2 - Het Score: 1.4072, Initial temp: 0.4948\n",
      "Client 3 - Het Score: 1.6845, Initial temp: 0.4307\n",
      "Client 4 - Het Score: 0.5174, Initial temp: 0.7721\n",
      "Client 5 - Het Score: 0.4435, Initial temp: 0.8011\n",
      "Client 6 - Het Score: 0.7640, Initial temp: 0.6825\n",
      "Client 7 - Het Score: 0.3035, Initial temp: 0.8592\n",
      "Client 8 - Het Score: 1.1237, Initial temp: 0.5701\n",
      "Client 9 - Het Score: 0.7487, Initial temp: 0.6878\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.5997, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.0684\n",
      "Epoch [2/5], Loss: 0.0249\n",
      "Epoch [3/5], Loss: 0.0187\n",
      "Epoch [4/5], Loss: 0.0120\n",
      "Epoch [5/5], Loss: 0.0085\n",
      "Client 0 Private Data Validation Accuracy: 58.12%\n",
      "\n",
      "--- Client 1 Local Training with T=0.7368, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.6781\n",
      "Epoch [2/5], Loss: 0.2262\n",
      "Epoch [3/5], Loss: 0.1214\n",
      "Epoch [4/5], Loss: 0.0804\n",
      "Epoch [5/5], Loss: 0.0532\n",
      "Client 1 Private Data Validation Accuracy: 65.62%\n",
      "\n",
      "--- Client 2 Local Training with T=0.4948, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3940\n",
      "Epoch [2/5], Loss: 0.0708\n",
      "Epoch [3/5], Loss: 0.0409\n",
      "Epoch [4/5], Loss: 0.0481\n",
      "Epoch [5/5], Loss: 0.0267\n",
      "Client 2 Private Data Validation Accuracy: 70.83%\n",
      "\n",
      "--- Client 3 Local Training with T=0.4307, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2918\n",
      "Epoch [2/5], Loss: 0.0459\n",
      "Epoch [3/5], Loss: 0.0142\n",
      "Epoch [4/5], Loss: 0.0098\n",
      "Epoch [5/5], Loss: 0.0061\n",
      "Client 3 Private Data Validation Accuracy: 90.10%\n",
      "\n",
      "--- Client 4 Local Training with T=0.7721, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1096\n",
      "Epoch [2/5], Loss: 0.0490\n",
      "Epoch [3/5], Loss: 0.0316\n",
      "Epoch [4/5], Loss: 0.0221\n",
      "Epoch [5/5], Loss: 0.0211\n",
      "Client 4 Private Data Validation Accuracy: 61.98%\n",
      "\n",
      "--- Client 5 Local Training with T=0.8011, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3017\n",
      "Epoch [2/5], Loss: 0.1686\n",
      "Epoch [3/5], Loss: 0.0840\n",
      "Epoch [4/5], Loss: 0.0489\n",
      "Epoch [5/5], Loss: 0.0476\n",
      "Client 5 Private Data Validation Accuracy: 63.28%\n",
      "\n",
      "--- Client 6 Local Training with T=0.6825, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.9028\n",
      "Epoch [2/5], Loss: 0.2878\n",
      "Epoch [3/5], Loss: 0.1340\n",
      "Epoch [4/5], Loss: 0.0630\n",
      "Epoch [5/5], Loss: 0.0360\n",
      "Client 6 Private Data Validation Accuracy: 68.75%\n",
      "\n",
      "--- Client 7 Local Training with T=0.8592, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1345\n",
      "Epoch [2/5], Loss: 0.0532\n",
      "Epoch [3/5], Loss: 0.0373\n",
      "Epoch [4/5], Loss: 0.0316\n",
      "Epoch [5/5], Loss: 0.0277\n",
      "Client 7 Private Data Validation Accuracy: 68.23%\n",
      "\n",
      "--- Client 8 Local Training with T=0.5701, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1845\n",
      "Epoch [2/5], Loss: 0.0289\n",
      "Epoch [3/5], Loss: 0.0187\n",
      "Epoch [4/5], Loss: 0.0136\n",
      "Epoch [5/5], Loss: 0.0166\n",
      "Client 8 Private Data Validation Accuracy: 82.42%\n",
      "\n",
      "--- Client 9 Local Training with T=0.6878, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2005\n",
      "Epoch [2/5], Loss: 0.0590\n",
      "Epoch [3/5], Loss: 0.0349\n",
      "Epoch [4/5], Loss: 0.0245\n",
      "Epoch [5/5], Loss: 0.0179\n",
      "Client 9 Private Data Validation Accuracy: 67.97%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 54.53%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 58.12%\n",
      "Client 1 Final Private Data Validation Accuracy: 67.19%\n",
      "Client 2 Final Private Data Validation Accuracy: 69.79%\n",
      "Client 3 Final Private Data Validation Accuracy: 89.58%\n",
      "Client 4 Final Private Data Validation Accuracy: 64.06%\n",
      "Client 5 Final Private Data Validation Accuracy: 60.16%\n",
      "Client 6 Final Private Data Validation Accuracy: 65.62%\n",
      "Client 7 Final Private Data Validation Accuracy: 69.27%\n",
      "Client 8 Final Private Data Validation Accuracy: 82.03%\n",
      "Client 9 Final Private Data Validation Accuracy: 72.66%\n",
      "\n",
      "Final Server Test Accuracy: 54.53%\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = js\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 0.2739, Initial temp: 0.8720\n",
      "Client 1 - Het Score: 0.1681, Initial temp: 0.9194\n",
      "Client 2 - Het Score: 0.3461, Initial temp: 0.8411\n",
      "Client 3 - Het Score: 0.4343, Initial temp: 0.8048\n",
      "Client 4 - Het Score: 0.1533, Initial temp: 0.9262\n",
      "Client 5 - Het Score: 0.1192, Initial temp: 0.9421\n",
      "Client 6 - Het Score: 0.1991, Initial temp: 0.9052\n",
      "Client 7 - Het Score: 0.0819, Initial temp: 0.9599\n",
      "Client 8 - Het Score: 0.2538, Initial temp: 0.8808\n",
      "Client 9 - Het Score: 0.1906, Initial temp: 0.9091\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.8720, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.0747\n",
      "Epoch [2/5], Loss: 0.0355\n",
      "Epoch [3/5], Loss: 0.0253\n",
      "Epoch [4/5], Loss: 0.0190\n",
      "Epoch [5/5], Loss: 0.0132\n",
      "Client 0 Private Data Validation Accuracy: 58.75%\n",
      "\n",
      "--- Client 1 Local Training with T=0.9194, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.6950\n",
      "Epoch [2/5], Loss: 0.3071\n",
      "Epoch [3/5], Loss: 0.1496\n",
      "Epoch [4/5], Loss: 0.0896\n",
      "Epoch [5/5], Loss: 0.0615\n",
      "Client 1 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Client 2 Local Training with T=0.8411, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.5909\n",
      "Epoch [2/5], Loss: 0.2030\n",
      "Epoch [3/5], Loss: 0.0799\n",
      "Epoch [4/5], Loss: 0.0581\n",
      "Epoch [5/5], Loss: 0.0453\n",
      "Client 2 Private Data Validation Accuracy: 72.40%\n",
      "\n",
      "--- Client 3 Local Training with T=0.8048, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3455\n",
      "Epoch [2/5], Loss: 0.0637\n",
      "Epoch [3/5], Loss: 0.0372\n",
      "Epoch [4/5], Loss: 0.0225\n",
      "Epoch [5/5], Loss: 0.0161\n",
      "Client 3 Private Data Validation Accuracy: 90.62%\n",
      "\n",
      "--- Client 4 Local Training with T=0.9262, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1077\n",
      "Epoch [2/5], Loss: 0.0540\n",
      "Epoch [3/5], Loss: 0.0332\n",
      "Epoch [4/5], Loss: 0.0235\n",
      "Epoch [5/5], Loss: 0.0228\n",
      "Client 4 Private Data Validation Accuracy: 58.33%\n",
      "\n",
      "--- Client 5 Local Training with T=0.9421, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2707\n",
      "Epoch [2/5], Loss: 0.1705\n",
      "Epoch [3/5], Loss: 0.0901\n",
      "Epoch [4/5], Loss: 0.0585\n",
      "Epoch [5/5], Loss: 0.0479\n",
      "Client 5 Private Data Validation Accuracy: 60.94%\n",
      "\n",
      "--- Client 6 Local Training with T=0.9052, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.9682\n",
      "Epoch [2/5], Loss: 0.5478\n",
      "Epoch [3/5], Loss: 0.2431\n",
      "Epoch [4/5], Loss: 0.1029\n",
      "Epoch [5/5], Loss: 0.0600\n",
      "Client 6 Private Data Validation Accuracy: 68.75%\n",
      "\n",
      "--- Client 7 Local Training with T=0.9599, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1046\n",
      "Epoch [2/5], Loss: 0.0468\n",
      "Epoch [3/5], Loss: 0.0327\n",
      "Epoch [4/5], Loss: 0.0264\n",
      "Epoch [5/5], Loss: 0.0238\n",
      "Client 7 Private Data Validation Accuracy: 69.79%\n",
      "\n",
      "--- Client 8 Local Training with T=0.8808, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2070\n",
      "Epoch [2/5], Loss: 0.0582\n",
      "Epoch [3/5], Loss: 0.0357\n",
      "Epoch [4/5], Loss: 0.0260\n",
      "Epoch [5/5], Loss: 0.0402\n",
      "Client 8 Private Data Validation Accuracy: 81.64%\n",
      "\n",
      "--- Client 9 Local Training with T=0.9091, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2756\n",
      "Epoch [2/5], Loss: 0.0792\n",
      "Epoch [3/5], Loss: 0.0472\n",
      "Epoch [4/5], Loss: 0.0337\n",
      "Epoch [5/5], Loss: 0.0224\n",
      "Client 9 Private Data Validation Accuracy: 71.88%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 54.66%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 58.75%\n",
      "Client 1 Final Private Data Validation Accuracy: 65.62%\n",
      "Client 2 Final Private Data Validation Accuracy: 71.35%\n",
      "Client 3 Final Private Data Validation Accuracy: 90.10%\n",
      "Client 4 Final Private Data Validation Accuracy: 58.33%\n",
      "Client 5 Final Private Data Validation Accuracy: 57.81%\n",
      "Client 6 Final Private Data Validation Accuracy: 65.62%\n",
      "Client 7 Final Private Data Validation Accuracy: 68.75%\n",
      "Client 8 Final Private Data Validation Accuracy: 81.25%\n",
      "Client 9 Final Private Data Validation Accuracy: 71.88%\n",
      "\n",
      "Final Server Test Accuracy: 54.66%\n",
      "\n",
      "\n",
      "########################################\n",
      "Running experiment with measure = entropy_diff\n",
      "########################################\n",
      "Data partitioning complete.\n",
      "Client 0 - Het Score: 0.8527, Initial temp: 0.6529\n",
      "Client 1 - Het Score: 0.7037, Initial temp: 0.7034\n",
      "Client 2 - Het Score: 1.3864, Initial temp: 0.5000\n",
      "Client 3 - Het Score: 1.6167, Initial temp: 0.4456\n",
      "Client 4 - Het Score: 0.5006, Initial temp: 0.7786\n",
      "Client 5 - Het Score: 0.4251, Initial temp: 0.8085\n",
      "Client 6 - Het Score: 0.9316, Initial temp: 0.6276\n",
      "Client 7 - Het Score: 0.3201, Initial temp: 0.8521\n",
      "Client 8 - Het Score: 1.0807, Initial temp: 0.5825\n",
      "Client 9 - Het Score: 0.8083, Initial temp: 0.6676\n",
      "\n",
      "==================== COMMUNICATION ROUND 1 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 2 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 3 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 4 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 5 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 6 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 7 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 8 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 9 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 10 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 11 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 12 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 13 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 14 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 15 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 16 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 17 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 18 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 19 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 20 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 21 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 22 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 23 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 24 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 25 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 26 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 27 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 28 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 29 ====================\n",
      "\n",
      "==================== COMMUNICATION ROUND 30 ====================\n",
      "\n",
      "--- LOCAL TRAINING OF CLIENTS ---\n",
      "\n",
      "--- Client 0 Local Training with T=0.6529, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.0740\n",
      "Epoch [2/5], Loss: 0.0282\n",
      "Epoch [3/5], Loss: 0.0189\n",
      "Epoch [4/5], Loss: 0.0144\n",
      "Epoch [5/5], Loss: 0.0110\n",
      "Client 0 Private Data Validation Accuracy: 63.75%\n",
      "\n",
      "--- Client 1 Local Training with T=0.7034, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.6248\n",
      "Epoch [2/5], Loss: 0.2482\n",
      "Epoch [3/5], Loss: 0.1192\n",
      "Epoch [4/5], Loss: 0.0705\n",
      "Epoch [5/5], Loss: 0.0540\n",
      "Client 1 Private Data Validation Accuracy: 68.75%\n",
      "\n",
      "--- Client 2 Local Training with T=0.5000, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3965\n",
      "Epoch [2/5], Loss: 0.1263\n",
      "Epoch [3/5], Loss: 0.0980\n",
      "Epoch [4/5], Loss: 0.1043\n",
      "Epoch [5/5], Loss: 0.0629\n",
      "Client 2 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Client 3 Local Training with T=0.4456, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3042\n",
      "Epoch [2/5], Loss: 0.0334\n",
      "Epoch [3/5], Loss: 0.0162\n",
      "Epoch [4/5], Loss: 0.0086\n",
      "Epoch [5/5], Loss: 0.0053\n",
      "Client 3 Private Data Validation Accuracy: 90.10%\n",
      "\n",
      "--- Client 4 Local Training with T=0.7786, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1167\n",
      "Epoch [2/5], Loss: 0.0483\n",
      "Epoch [3/5], Loss: 0.0342\n",
      "Epoch [4/5], Loss: 0.0239\n",
      "Epoch [5/5], Loss: 0.0219\n",
      "Client 4 Private Data Validation Accuracy: 61.46%\n",
      "\n",
      "--- Client 5 Local Training with T=0.8085, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.3013\n",
      "Epoch [2/5], Loss: 0.1586\n",
      "Epoch [3/5], Loss: 0.0816\n",
      "Epoch [4/5], Loss: 0.0503\n",
      "Epoch [5/5], Loss: 0.0487\n",
      "Client 5 Private Data Validation Accuracy: 58.59%\n",
      "\n",
      "--- Client 6 Local Training with T=0.6276, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.9331\n",
      "Epoch [2/5], Loss: 0.3451\n",
      "Epoch [3/5], Loss: 0.1484\n",
      "Epoch [4/5], Loss: 0.0749\n",
      "Epoch [5/5], Loss: 0.0301\n",
      "Client 6 Private Data Validation Accuracy: 71.88%\n",
      "\n",
      "--- Client 7 Local Training with T=0.8521, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1254\n",
      "Epoch [2/5], Loss: 0.0540\n",
      "Epoch [3/5], Loss: 0.0374\n",
      "Epoch [4/5], Loss: 0.0302\n",
      "Epoch [5/5], Loss: 0.0265\n",
      "Client 7 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Client 8 Local Training with T=0.5825, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.1585\n",
      "Epoch [2/5], Loss: 0.0303\n",
      "Epoch [3/5], Loss: 0.0224\n",
      "Epoch [4/5], Loss: 0.0162\n",
      "Epoch [5/5], Loss: 0.0192\n",
      "Client 8 Private Data Validation Accuracy: 78.91%\n",
      "\n",
      "--- Client 9 Local Training with T=0.6676, LR=0.010000 ---\n",
      "Epoch [1/5], Loss: 0.2730\n",
      "Epoch [2/5], Loss: 0.0570\n",
      "Epoch [3/5], Loss: 0.0371\n",
      "Epoch [4/5], Loss: 0.0260\n",
      "Epoch [5/5], Loss: 0.0156\n",
      "Client 9 Private Data Validation Accuracy: 70.31%\n",
      "\n",
      "--- Server Evaluation ---\n",
      "Server Test Data Validation Accuracy: 55.18%\n",
      "\n",
      "--- Clients Final Evaluation on Private Data sets ---\n",
      "Client 0 Final Private Data Validation Accuracy: 63.75%\n",
      "Client 1 Final Private Data Validation Accuracy: 70.31%\n",
      "Client 2 Final Private Data Validation Accuracy: 70.31%\n",
      "Client 3 Final Private Data Validation Accuracy: 90.10%\n",
      "Client 4 Final Private Data Validation Accuracy: 61.98%\n",
      "Client 5 Final Private Data Validation Accuracy: 55.47%\n",
      "Client 6 Final Private Data Validation Accuracy: 67.19%\n",
      "Client 7 Final Private Data Validation Accuracy: 69.27%\n",
      "Client 8 Final Private Data Validation Accuracy: 78.52%\n",
      "Client 9 Final Private Data Validation Accuracy: 71.88%\n",
      "\n",
      "Final Server Test Accuracy: 55.18%\n"
     ]
    },
    {
     "data": {
      "image/png": "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AQMG5Lrc7Xabtm3bGsuyTP/+/c348eNNly5djCQzaNAgr3WHDx9uJJmrrrrKvPTSS+a1114zt99+u3niiSc863Tt2tX06NHDvPTSS2bSpEnm1ltvNZLM448/7tP+2NjY8+ZvjDFDhgwxlmV5jZO8cLlcJjQ01DRq1MgcPXr0vOv6O/5y+5vQr18/ExUVZdLS0rzWnzFjhpFkfvrpp3x7rXP5/vvvjSTz9ddfG2OM6devn6lTp47PegsWLDBBQUEmNjbWDBs2zEyaNMk89NBDpn379p511q1bZyIiIkyZMmXMkCFDzJtvvmkGDx5s6tat61knt8806++Fv+3x92/y+frjX533UPxQJKHISE5ONpJ8JvPcrF271kgy/fv394o//vjjRpL56quvPLHY2FgjyXz33Xee2JdffmkkmdDQULNr1y5P/M033/T58pP1BeTBBx/0xNxut+ncubMJCgoyhw8f9sRPnjzplU96erq54oorTNu2bb3ikozD4TAbNmzwadvZRVJkZOQ5v5ikp6eb8uXLmyuuuMKcOnXKE//vf/9rJJlnnnnGpy1nT+gNGjQwjRo1yvU1jDHm0KFDJigoyFx77bVeReT48eONJDN16lRPLOsPmf29yU3Wuuf6Zy+Svv76ayPJfPDBB17b+eKLL3zil19+uWnVqpXPaz777LMmPDzcbN261Sv+73//2zidTrN7925jTHaRFBERYQ4dOuS1bv369U358uW9viitW7fOOBwO07t3b0+sS5cuJiwszPz++++e2LZt20xAQIDXH/udO3cap9Npnn/+ea/X+eWXX0xAQIBXvFWrVkaSeffddz2xtLQ0U7FiRdO9e3dP7L333jMOh8PzxSbL5MmTjSTz7bffemL2IskYY2bNmuUzFow582UsOjra3HbbbV7xV1991ViWZbZv327OJatIMsaYW265xbRr186z3YoVK5oRI0bkWCT5+5kZ4984HDt27Hn7aF6LJElm8uTJXuv6+xn4k09OLmQuXL58uSd26NAhExwcbB577LHzvtb5iqS5c+caSea5557zit9yyy3Gsizz66+/GmPO9H+Hw2G6devmNZcYc2ZuzXL252iMMf/4xz9MWFiYOX36tCfmb5GUkZFh7rzzThMUFGTCw8O9/ibkxTPPPGMkmfDwcHP99deb559/3qxatcpnvbyMv9z+JmT9rTr7y3mnTp28dirmx2udy8CBA01MTIzn81mwYIGR5FVcZWZmmurVq5vY2FivnTrGeH+uLVu2NCVLlvT623v2OnktknJrjz9zgT/98a/Oeyh+ON0ORUZKSookqWTJkn6t/7///U+S9Oijj3rFH3vsMUnyuXapTp06at68uefxlVdeKUlq27atqlat6hPfvn27z2va7+qUdXg/PT1dixYt8sRDQ0M9///nn38qOTlZ11xzjc+pcZLUqlUr1alT5zwtPXNdzw8//KB9+/bluHzlypU6dOiQHnjgAa9zsDt37qz4+Pgcr+P65z//6fX4mmuuybHNdosWLVJ6eroGDRrkdYH/vffeq4iIiL98vdjHH3+shQsX+vyrUKGC13qzZs1SZGSkOnTooCNHjnj+NWrUSCVKlNCSJUvO+1qzZs3SNddco1KlSnlto3379nK5XFq+fLnX+t27d1e5cuU8j/fv36+1a9eqT58+Kl26tCder149dejQwdM/XS6XFi1apK5du6py5cqe9WrWrKnrr7/e6zXmzJkjt9utHj16eOVUsWJF1apVy6ddJUqU0J133ul5HBQUpKZNm3p9jrNmzVJCQoLi4+O9ttm2bVtJ8uu9OpvD4dAdd9yhzz77TMePH/fEP/jgA1111VWqXr2639u6/fbbtXTpUh04cEBfffWVDhw4kOupdnn5zPwZh1nXy3366ac+p+tdqODgYJ9To/z9DC40nwuZC6+55hrP43Llyql27drnHf/+5uJ0OvXQQw/55GKM0fz58yVJc+fOldvt1jPPPOM1l0jyOpXK/jkeP35cR44c0TXXXKOTJ09q8+bNec5v8ODBmj9/vn755RddeeWV6tSpk9auXetZvn//flmW5XW9Uk5GjBihpKQkNWjQQF9++aWefPJJNWrUSA0bNvQ63Tav4y+nvwlt27ZV2bJlNXPmTE/szz//1MKFC3Xbbbfl62vlJjMzUzNnztRtt93m+Xzatm2r8uXL64MPPvCst2bNGu3YsUODBg3yuR4163mHDx/W8uXL1a9fP6+/vfZ1LkRu7fFnLvCnP+bnvIfigRs3oMiIiIiQJK/J51x27dolh8Phc+e0ihUrKioqSrt27fKKnz0ZR0ZGSpJiYmJyjNuvbZHOTJA1atTwil122WWS5HW9wn//+18999xzWrt2rdd54DlN/v5Oqi+++KLuvvtuxcTEqFGjRurUqZN69+7tySerrTmdEx0fH69vvvnGKxYSEuL1hV+SSpUq5dPms+X2OkFBQapRo4bPe55XLVu2VNmyZX3iZ198u23bNiUnJ/uc+5/Fnwunt23bpp9//tnnfchtG2d/Vud6zxMSEvTll18qNTVVKSkpOnXqVI53+Ds7tm3bNhljVKtWrRxzCgwM9HocHR3t069KlSqln3/+2WubmzZt8rud/urdu7fGjBmjTz75RL1799aWLVu0atUqTZ48OU/b6dSpk0qWLKmZM2dq7dq1atKkiWrWrJnjbcfz8pn5Mw5vu+02vfPOO+rfv7/+/e9/q127drr55pt1yy23+HxR8leVKlV8bujh72dwofn81blQ8m/8+2PXrl2qXLmyz86uhIQEz3LpzM88OByO835J37Bhg5566il99dVXnh1pWZKTk/OU2++//67XX39do0aN0mWXXaa5c+eqVatWuvbaa/X111+rdu3aWr9+vaTsnWXn0qtXL/Xq1UspKSn64YcfNH36dCUlJalLly5av369QkJC8jz+cvqbEBAQoO7duyspKUlpaWkKDg7WnDlzlJGR4VUk5cdr5WbBggU6fPiwmjZtql9//dUTb9OmjT788EONGTNGDodDv/32myTpiiuuyHVbWcX4uda5ELm1x5+5wN/+mF/zHooHiiQUGREREapcubLnj5S//N3z5HQ68xQ3OVxkfD5ff/21brzxRrVs2VITJ05UpUqVFBgYqGnTpuV4gat9D9e59OjRQ9dcc40++eQTLViwQC+99JLGjBmjOXPm+ByN8EdubS4u3G63zx5Mu9y+JJy9jQ4dOmjw4ME5Ls8qgLP4+1n9FW63W5Zlaf78+Tl+RiVKlPB67E/fdbvdqlu3rl599dUc1z17J4G/6tSpo0aNGun9999X79699f777ysoKMjnTnXnExwcrJtvvlkzZszQ9u3bz3mDD38/M3/HYWhoqJYvX64lS5Zo3rx5+uKLLzRz5ky1bdtWCxYskNPpzHV+ye3GATn1E38/A3/yOZe/OhdeyJx3MR07dkytWrVSRESERo4cqbi4OIWEhGj16tV64okn8nz074cffpDL5VKzZs0knTlrYf78+WrRooXat2+vr7/+Wm+99ZYSExPz9AU+IiJCHTp0UIcOHRQYGKgZM2bohx9+UKtWrfI8/nKbZ3r27Kk333xT8+fPV9euXfWf//xH8fHxSkxM9KyTX6+Vk6y5NrfxvWzZMrVp08bv7fkjP8ZeXv8mn09+zXsoHiiSUKTccMMNeuutt7RixQqvU+NyEhsbK7fbrW3btnn2UkrSwYMHdezYMcXGxuZrbm63W9u3b/f68rx161ZJ8tyB5+OPP1ZISIi+/PJLr7slTZs27S+/fqVKlfTAAw/ogQce0KFDh9SwYUM9//zzuv766z1t3bJli+fUiixbtmzJt/fC/jr2o2rp6enasWOH2rdvny+vcz5xcXFatGiRWrRocd4/9Ln9oY2Li9OJEycuOGf7e3G2zZs3q2zZsgoPD/fc0t2+9zXL2bG4uDgZY1S9enWfIu1CxcXFad26dWrXrl2eT2U53/q9e/fWo48+qv3793tu212qVKk853j77bdr6tSpcjgc6tmzZ67r+fuZ5WUcOhwOtWvXTu3atdOrr76qF154QU8++aSWLFmi9u3be9pjvwOfpDwdNc3LZ3C+fHJS0HPhucTGxmrRokU6fvy419GkrFPjsnKJi4uT2+3Wxo0bVb9+/Ry3tXTpUh09elRz5sxRy5YtPfEdO3ZcUG5Z7739TpEVKlTQl19+qRYtWqhVq1bau3ev5syZc0Hbl6TGjRtrxowZ2r9/v6S/Nv7sWrZsqUqVKmnmzJm6+uqr9dVXX3ndCS4/X+tsqamp+vTTT3Xbbbfplltu8Vn+0EMP6YMPPlCbNm0UFxcnSVq/fn2u/TXrb8f5doiWKlXKZ9xJeRt7/s4F/vTHLPk176Ho45okFCmDBw9WeHi4+vfvr4MHD/os/+233zy3Ee3UqZMkady4cV7rZO1F69y5c77nN378eM//G2M0fvx4BQYGql27dpLk2fNs39O1c+dOzZ0794Jf0+Vy+ZxWUr58eVWuXNlz6kDjxo1Vvnx5TZ482et0gvnz52vTpk359l60b99eQUFBev311732Ok+ZMkXJyckX5T3PSY8ePeRyufTss8/6LMvMzPT6wxoeHp7jH9oePXpoxYoV+vLLL32WHTt2TJmZmefMoVKlSqpfv75mzJjhtf3169drwYIFnv7pdDrVvn17zZ071+uasl9//dVzfUaWm2++WU6nUyNGjPDZq2+M0dGjR8+ZU0569Oih33//XW+//bbPslOnTik1NTXX54aHh0vyLRCy9OrVS5Zl6eGHH9b27du9ro/KizZt2ujZZ5/V+PHjVbFixVzX8/cz83cc/vHHHz7byfqClDWOsr702a93crlcefpBa38/A3/yyUlhzIXnysXlcnnNlZI0duxYWZblOfLdtWtXORwOjRw50ueIUFbfzzriZR8L6enpmjhx4gXldvXVVys4OFijR4/WyZMnPfG4uDiNGzdOu3fvVmRkpFq1anXO7Zw8eVIrVqzIcVnWmM46DfevjD87h8OhW265RZ9//rnee+89ZWZmep1ql5+vdbZPPvlEqampGjBggG655RaffzfccIM+/vhjpaWlqWHDhqpevbrGjRvnM29kfY7lypVTy5YtNXXqVO3evTvHdaQzn0tycrLX6cP79+/XJ5984nfu/s4F/vTHLPk176Ho40gSipS4uDglJSXptttuU0JCgnr37q0rrrhC6enp+u677zRr1izP77gkJibq7rvv1ltvveU5LePHH3/UjBkz1LVr13w/9B8SEqIvvvhCd999t6688krNnz9f8+bN09ChQz2nd3Xu3FmvvvqqrrvuOt1+++06dOiQJkyYoJo1a3pN9Hlx/PhxRUdH65ZbblFiYqJKlCihRYsW6aefftIrr7wi6cy1KmPGjFHfvn3VqlUr9erVSwcPHtRrr72matWq6ZFHHsmX96BcuXIaMmSIRowYoeuuu0433nijtmzZookTJ6pJkyYF9seiVatW+sc//qFRo0Zp7dq1uvbaaxUYGKht27Zp1qxZeu211zx7PBs1aqRJkybpueeeU82aNVW+fHm1bdtW//rXv/TZZ5/phhtuUJ8+fdSoUSOlpqbql19+0ezZs7Vz584cr4+ye+mll3T99derefPmuueee3Tq1Cm98cYbioyM9DptbPjw4VqwYIFatGih+++/3/Ml8oorrvC6aDwuLk7PPfechgwZop07d6pr164qWbKkduzYoU8++UT33Xef12+i+OOuu+7Sf/7zH/3zn//UkiVL1KJFC7lcLm3evFn/+c9/9OWXX6px48Y5Prd+/fpyOp0aM2aMkpOTFRwc7LlYWzrTH6677jrNmjVLUVFRF/xl3OFw6Kmnnjrvev5+Zv6Ow5EjR2r58uXq3LmzYmNjdejQIU2cOFHR0dG6+uqrJUmXX365mjVrpiFDhuiPP/5Q6dKl9dFHH523iLbz9zPwJ5+cFPRcuHLlSj333HM+8datW6tLly5q06aNnnzySe3c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gCZOnKi7775b06dPz6csL8z06dM1aNCgYlGkjBo1Sk899ZRGjx6tf/3rX4WdziWBI0kAAAAoUDExMfroo4906tQpT+z06dNKSkpS1apVCzGz4mnq1KkaPHiwpk6dWtipKD09vbBTyBcUSQAAAChQDRs2VExMjObMmeOJzZkzR1WrVlWDBg281nW73Ro1apSqV6+u0NBQJSYmavbs2Z7lLpdL99xzj2d57dq19dprr3lto0+fPuratatefvllVapUSWXKlNGAAQP+0t3Pdu/erZtuukklSpRQRESEevTooYMHD3qt8/nnn6tJkyYKCQlR2bJl1a1bN8+y9957T40bN1bJkiVVsWJF3X777Z7f98mLZcuW6dSpUxo5cqRSUlL03XffeS13u9168cUXVbNmTQUHB6tq1ap6/vnnPcv37t2rXr16qXTp0goPD1fjxo31ww8/SMp+3+wGDRqk1q1bex63bt1aAwcO1KBBg1S2bFl17NhRkvTqq6+qbt26Cg8PV0xMjB544AGdOHHCa1vffvutWrdurbCwMJUqVUodO3bUn3/+qXfffVdlypRRWlqa1/pdu3bVXXfdlef36EIUapFUrVo1WZbl82/AgAGSzuxRGDBggMqUKaMSJUqoe/fuPp0PAAAAxU+/fv00bdo0z+OpU6eqb9++PuuNGjVK7777riZPnqwNGzbokUce0Z133qlly5ZJOlMEREdHa9asWdq4caOeeeYZDR06VP/5z3+8trNkyRL99ttvWrJkiWbMmKHp06df8Cl9brdbN910k/744w8tW7ZMCxcu1Pbt23Xbbbd51pk3b566deumTp06ac2aNVq8eLGaNm3qWZ6RkaFnn31W69at09y5c7Vz50716dMnz7lMmTJFvXr1UmBgoHr16qUpU6Z4LR8yZIhGjx6tp59+Whs3blRSUpIqVKggSTpx4oRatWql33//XZ999pnWrVunwYMHy+125ymHGTNmKCgoSN9++63nFEqHw6HXX39dGzZs0IwZM/TVV19p8ODBnuesXbtW7dq1U506dbRixQp988036tKli1wul2699Va5XC599tlnnvUPHTqkefPmqV+/fnl+jy6EZYwxBfJKOTh8+LBcLpfn8fr169WhQwctWbJErVu31v3336958+Zp+vTpioyM1MCBA+VwOPTtt9/6/RopKSmKjIxUcnKyIiIi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+vzLXust+fJ/uu36VmpSs6I6taivyWPHKDMz07PcsixNmjRJN954o8LDw/Xss88qOjpakyZN8trOmjVr5HA4tGvXmaNwx44dU//+/VWuXDlFRESobdu2WrdunWf94cOHq379+nrnnXdUvXp1hYSEnLM9U6ZM0e2336677rpLU6dO9Vm+d+9e9erVS6VLl1Z4eLgaN26sH374wbP8888/V5MmTRQSEqKyZcuqW7duXm2cO3eu1/aioqI0ffp0SdLOnTtlWZZmzpypVq1aKSQkRB988IGOHj2qXr16qUqVKgoLC1PdunX14YfeRxPdbrdefPFF1axZU8HBwapataqef/55SVLbtm01cOBAr/UPHz6soKAgLV68+Jzvx4WiSAIAAECR9e5bE3R5vfqaOX+ZevS+R88PfUw7f9vmWR4eXlLPvjpBc776XoOHj9KcD9/V++9M9Hv7ZcqV129bN2vb5o25rrP6h+/01CP/1B39/qlPFn+vp0eN1aezkvTOG694rTd8+HB169ZNv/zyi/r3769evXopKSnJa50PPvhALVq0UGxsrCTp1ltv1aFDhzR//nytWrVKDRs2VLt27fTHH394nvPrr7/q448/1pw5c7R27dpc8zx+/LhmzZqlO++8Ux06dFBycrK+/vprz/ITJ06oVatW+v333/XZZ59p3bp1Gjx4sNxutyRp3rx56tatmzp16qQ1a9Zo8eLFatq0qd/vZZZ///vfevjhh7Vp0yZ17NhRp0+fVqNGjTRv3jytX79e9913n+666y79+OOPnucMGTJEo0eP1tNPP62NGzcqKSlJFSpUkCT1799fSUlJSktL86z//vvvq0qVKmrbtm2e8/MHp9sBAACgQC1f/KWa1Y72PL66TXu9PHl6jute3baDbru7vySp3wOD9P47k/Tjd1+rWlwtSdJ9Dz/uWbdKTFXt+u1XffHZHPW9/2G/cunV9z6t+XGFbunQQpWjY1S3QWM1b9lWnbvdqqDgYEnS5HEvqt8Dg3Tjrb0kSdGx1TTg8aEa9/xwTXwl+7ql22+/XX379vU8vuOOO/TKK69o9+7dqlq1qtxutz766CM99dRTks6cGvfjjz/q0KFDCv7/13r55Zc1d+5czZ49W/fdd5+kM6fYvfvuuypXrtw52/LRRx+pVq1auvzyyyVJPXv21JQpU3TNNWduXJGUlKTDhw/r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AAAAAFDg/C6Sdu/erR49eqhu3bq64447VKtWLa1atUphYWFKTEzU/PnzL2aeAAAAAFAg/C6SevfuLYfDoZdeeknly5fXP/7xDwUFBWnEiBGaO3euRo0apR49elzMXAEAAADgovP7mqSVK1dq3bp1iouLU8eOHVW9enXPsoSEBC1fvlxvvfXWRUkSAAAAAAqK30VSo0aN9Mwzz+juu+/WokWLVLduXZ917rvvvnxNDgAAAAAKmt+n27377rtKS0vTI488ot9//11vvvnmxcwLAAAAAAqF30eSYmNjNXv27IuZCwAAAAAUOr+OJKWmpuZpo3ldHwAAAACKCr+KpJo1a2r06NHav39/rusYY7Rw4UJdf/31ev311/OcyOjRo2VZlgYNGuSJnT59WgMGDFCZMmVUokQJde/eXQcPHszztgEAAADAX36dbrd06VINHTpUw4cPV2Jioho3bqzKlSsrJCREf/75pzZu3KgVK1YoICBAQ4YM0T/+8Y88JfHTTz/pzTffVL169bzijzzyiObNm6dZs2YpMjJSAwcO1M0336xvv/02T9sHAAAAAH/5VSTVrl1bH3/8sXbv3q1Zs2bp66+/1nfffadTp06pbNmyatCggd5++21df/31cjqdeUrgxIkTuuOOO/T222/rueee88STk5M1ZcoUJSUlqW3btpKkadOmKSEhQd9//72aNWuWp9cBAAAAAH/4feMGSapataoee+wxPfbYY/mWwIABA9S5c2e1b9/eq0hatWqVMjIy1L59e08sPj5eVatW1YoVK3ItktLS0pSWluZ5nJKSIknKzMxUZmamJMnhcMjhcMjtdsvtdnvWzYq7XC4ZY84bdzqdsizLs117XJJcLpdP3JJRwFknOWa4LZ+4MVKmseSQkTOnuGXktLLjbiO5jCWnZeSwxV1GchtLAZaRZY+7Jbd845luychSoCO7ndlxKdAnd8mS8r1NmVaQHHLJYVxyW065lV18O4xLDrnksgJlZNnimXLI7RN3mgxZMsq0grxydJoMSUYun3i6JEsuK9ArHmDSZc6KWzJymgy55ZDbCsgh7pTbsuWe323KQ9/LKR4QECBjjFfcsiw5nU6f8ZFbPKfxFOgwxbbvSQU3nnz7ZDHqewU5nlwuv/uePc5cXvjjibnczzYVwbk8q+8U174nMZefiReh8VRE5vKzl+cmT0VSfvvo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/+XXAQAAAICc/KXfSapdu7ZefPFF7d27Vx9++GGenz9kyBAtX75cO3fu1C+//KIhQ4Zo6dKluuOOOxQZGal77rlHjz76qJYsWaJVq1apb9++at68OXe2AwAAAHDR5OkW4LlxOp3q2rWrunbtmqfnHTp0SL1799b+/fsVGRmpevXq6csvv1SHDmcu8ho7dqwcDoe6d++utLQ0dezYURMnTsyPlAEAAAAgR/lSJF2oKVOmnHN5SEiIJkyYoAkTJhRQRgAAAAD+7v7S6XYAAAAAcKmhSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAG4okAAAAALChSAIAAAAAm0ItkkaNGqUmTZqoZMmSKl++vLp27aotW7Z4rXP69GkNGDBAZcqUUYkSJdS9e3cdPHiwkDIGAAAAcKkr1CJp2bJlGjBggL7//nstXLhQGRkZuvbaa5WamupZ55FHHtHnn3+uWbNmadmyZdq3b59uvvnmQswaAAAAwKUsoDBf/IsvvvB6PH36dJUvX16rVq1Sy5YtlZycrClTpigpKUlt27aVJE2bNk0JCQn6/vvv1axZs8JIGwAAAMAlrFCLpLMlJydLkkqXLi1JWrVqlTIyMtS+fXvPOvHx8apatapWrFiRY5GUlpamtLQ0z+OUlBRJUmZmpjIzMyVJDodDDodDbrdbbrfbs25W3OVyyRhz3rjT6ZRlWZ7t2uOS5HK5fOKWjALOOn6X4bZ84sZImcaSQ0bOnOKW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SQAAAABgQ5GEYqFatWoaN25cYacBXLDWrVtr0KBBOS7r06ePunbtWqD5ABcDfRnIf2d/B7IsS3PnzvU83rx5s5o1a6aQkBDVr18/1xjypsjf3Q5/L9OnT9egQYN07Ngxr/hPP/2k8PDwwkkKAOCX1157TdwPCkXJ8OHDNXfuXK1du7awU8k3+/fvV6lSpTyPhw0bpvDwcG3ZskUlSpTINYa8oUhCgUlPT1dQUNAFPbdcuXL5nA0AIL/58yv2QFGUkZGhwMDAwk7DLxUrVvR6/Ntvv6lz586KjY09Zwx5w+l2uGhat26tgQMHatCgQSpbtqw6duyoV199VXXr1lV4eLhiYmL0wAMP6MSJE5KkpUuXqm/fvkpOTpZlWbIsS8OHD5fke6h59+7duummm1SiRAlFRESoR48eOnjwYCG0Ergw8+bNU2RkpD744IPCTgXIN/bT7WbPnq26desqNDRUZcqUUfv27ZWamlq4CaLYcbvdGjVqlKpXr67Q0FAlJiZq9uzZks58b7AsS4sXL1bjxo0VFhamq666Slu2bJF05uyUESNGaN26dZ7vFdOnT5d05pS1SZMm6cYbb1R4eLief/55SdKkSZMUFxenoKAg1a5dW++9955XPlnPu/766xUaGqoaNWp48pGktm3bauDAgV7POXz4sIKCgrR48eLztvfQoUPq0qWLQkNDVb169Rz/RthPt7MsS6tWrdLIkSM935tyiiHvKJJwUc2YMUNBQUH69ttvNXnyZDkcDr3++uvasGGDZsyYoa+++kqDBw+WJF111VUaN26cIiIitH//fu3fv1+PP/64zzbdbrduuukm/fHHH1q2bJkWLlyo7du367bbbivo5gEXJCkpSb169dIHH3ygO+64o7DTAfLd/v371atXL/Xr10+bNm3S0qVLdfPNN3MqHvJs1KhRevfddzV58mRt2LBBjzzyiO68804tW7bMs86TTz6pV155RStXrlRAQID69esnSbrtttv02GOP6fLLL/d8r7B/Vxg+fLi6deumX375Rf369dMnn3yihx9+WI899pjWr1+vf/zjH+rbt6+WLFnildPTTz+t7t27a926dbrjjjvUs2dPbdq0SZLUv39/JSUlKS0tzbP++++/rypVqqht27bnbW+fPn20Z88eLVmyRLNnz9bEiRN16NChXNffv3+/Lr/8cj322GOe7005xXABDHCRtGrVyjRo0OCc68yaNcuUKVPG83jatGkmMjLSZ73Y2FgzduxYY4wxCxYsME6n0+zevduzfMOGDUaS+fHHH/MldyC/tWrVyjz88MNm/PjxJjIy0ixdutSz7O677zY33XRT4SUH5JOsvrxq1SojyezcubOwU0Ixdvr0aRMWFma+++47r/g999xjevXqZZYsWWIkmUWLFnmWzZs3z0gyp06dMsYYM2zYMJOYmOizbUlm0KBBXrGrrrrK3HvvvV6xW2+91XTq1Mnref/85z+91rnyyivN/fffb4wx5tSpU6ZUqVJm5syZnuX16tUzw4cPP297t2zZ4vNdZtOmTUaS5ztQVg6ffPKJ53FiYqIZNmyY17ZyiiFvOJKEi6pRo0ZejxctWqR27dqpSpUqKlmypO666y4dPXpUJ0+e9HubmzZtUkxMjGJiYjyxOnXqKCoqyrMnByiKZs+erUceeUQLFy5Uq1atCjsd4KJJTExUu3btVLduXd166616++239eeffxZ2Wihmfv31V508eVIdOnRQiRIlPP/effdd/fbbb5716tWr5/n/SpUqSdI5j75kady4sdfjTZs2qUWLFl6xFi1a+Hy3aN68uc/jrHVCQkJ01113aerUqZKk1atXa/369erTp89589m0aZMCAgK8vjvFx8crKirqvM9F/qNIwkVlvyPdzp07dcMNN6hevXr6+OOPtWrVKk2YMEHSmZs6AJe6Bg0aqFy5cpo6dSqnHeGS5nQ6tXDhQs2fP1916tTRG2+8odq1a2vHjh2FnRqKkaxrlufNm6e1a9d6/m3cuNHrOiD7DRcsy5J05tT887lYd83t37+/Fi5cqL1792ratGlq27YtN1AohiiSUGBWrVolt9utV155Rc2aNdNll12mffv2ea0TFBQkl8t1zu0kJCRoz5492rNnjye2ceNGHTt2THXq1LkouQP5IS4uTkuWLNGnn36qBx98sLDTAS4qy7LUokULjRgxQmvWrFFQUJA++eSTwk4LxUidOnUUHBys3bt3q2bNml7/7GeTnIs/3yuyJCQk6Ntvv/WKffvttz7fLb7//nufxwkJCZ7HdevWVePGjfX2228rKSnJc43U+cTHxyszM1OrVq3yxLZs2eLzsygoGNwCHAWmZs2aysjI0BtvvKEuXbp4buZgV61aNZ04cUKLFy9WYmKiwsLCFBYW5rVO+/btVbduXd1xxx0aN26cMjMz9cD/tXf3IKm+YRzHf9IDFaHQoFA09IYSTxSFQ4SkgcRTUy9QuEfJoaGoNhuiQYoKmqIhhEgsgiKirRcoJMylIVprCYKmpoai/9kOejpxhDx5zp/vB57J65br3p4f1633t2/y+/3vRufA38btduv09FSBQECGYXBJMv6XUqmUjo+P1dXVJZfLpVQqpcfHx6wXSeB37Ha7pqamNDExobe3N/l8Pj09PSmZTMrhcOQ0namurtbt7a2urq5UVVUlu92u4uLiX9ZOT09rcHBQLS0tCgaDOjg40O7uro6OjrLqdnZ25PV65fP5FI/HdXl5qfX19aya4eFhjY2NqaysTH19fTnt1+PxyLIsjY6OanV1VYZhaHx8XKWlpTmtR34xScKXaW5u1vLysubn59XY2Kh4PK5oNJpV097ernA4rKGhITmdTi0sLLz7HpvNpv39fZWXl6ujo0PBYFC1tbXa3t7+qq0An+LxeHRycqJEIqHJyclCtwPkncPh0NnZmXp6euR2uxWJRLS0tKTu7u5Ct4Z/zNzcnGZmZhSNRtXQ0CDLsnR4eKiampqc1g8MDMiyLHV2dsrpdCqRSHxY29vbq5WVFS0uLso0Ta2trSkWiykQCGTVzc7OamtrS01NTdrY2FAikXg3bQqFQjIMQ6FQSCUlJTnvNxaLqbKyUn6/X/39/RoZGZHL5cp5PfLH9h8H4wEAQB6EQiEVFRVpc3Oz0K0Af4TNZtPe3t6P+8A+cnd3p7q6OqXTabW2tn5Nc8grJkkAAOBTXl9fdXNzo4uLC5mmWeh2gIJ5eXnRw8ODIpGI2traCEj/MEISAAD4lOvra3m9XpmmqXA4XOh2gIJJJpOqqKhQOp1+97vr8/PzrL8y//nB34XjdgAAAMAf9vz8rPv7+w8/r6+v/8Ju8DuEJAAAAADIwHE7AAAAAMhASAIAAACADIQkAAAAAMhASAIAAACADIQkAAAAAMhASAIAAACADIQkAAAAAMhASAIAAACADN8BlND31FpmYHIAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "BATCH_SIZE = 64\n",
    "LEARNING_RATE = 0.01\n",
    "LOCAL_EPOCHS = 5\n",
    "NUM_OF_CLIENTS = 10\n",
    "COMM_ROUND = 30\n",
    "ALPHA = 0.5  \n",
    "FRAC = 0.1   \n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "def run_all_measures():\n",
    "    measures = [\"ratio\", \"kl\", \"js\", \"entropy_diff\"]\n",
    "    \n",
    "    results = {\n",
    "        \"measure\": [],\n",
    "        \"mean_local_acc\": [],\n",
    "        \"final_server_acc\": []\n",
    "    }\n",
    "    \n",
    "    for m in measures:\n",
    "        client_accs, server_acc, _ = run_fedchill_het_ti(metric=m)\n",
    "        \n",
    "        \n",
    "        mean_local = np.mean(client_accs)\n",
    "        \n",
    "        results[\"measure\"].append(m)\n",
    "        results[\"mean_local_acc\"].append(mean_local)\n",
    "        results[\"final_server_acc\"].append(server_acc)\n",
    "    \n",
    "    return results\n",
    "\n",
    "\n",
    "def plot_results(results):\n",
    "    measures = results[\"measure\"]\n",
    "    x = np.arange(len(measures))\n",
    "    \n",
    "    plt.figure(figsize=(10,6))\n",
    "    \n",
    "    \n",
    "    plt.bar(x - 0.2, results[\"mean_local_acc\"], width=0.4, label=\"Mean Local Accuracy\")\n",
    "    \n",
    "    \n",
    "    plt.bar(x + 0.2, results[\"final_server_acc\"], width=0.4, label=\"Final Server Accuracy\")\n",
    "    \n",
    "    plt.xticks(x, measures)\n",
    "    plt.ylabel(\"Accuracy (%)\")\n",
    "    plt.title(\"Comparison of Heterogeneity Measures on Local & Server Accuracy\")\n",
    "    plt.legend()\n",
    "    plt.grid(axis=\"y\", linestyle=\"--\", alpha=0.7)\n",
    "    plt.show()\n",
    "\n",
    "\n",
    "\n",
    "results4 = run_all_measures()\n",
    "plot_results(results4)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2025-09-16T18:31:30.645055Z",
     "iopub.status.busy": "2025-09-16T18:31:30.644735Z",
     "iopub.status.idle": "2025-09-16T18:31:30.649552Z",
     "shell.execute_reply": "2025-09-16T18:31:30.648973Z",
     "shell.execute_reply.started": "2025-09-16T18:31:30.645030Z"
    },
    "trusted": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'measure': ['ratio', 'kl', 'js', 'entropy_diff'], 'mean_local_acc': [68.91927083333334, 69.84895833333333, 68.94791666666667, 69.87760416666666], 'final_server_acc': [54.89, 54.53, 54.66, 55.18]}\n"
     ]
    }
   ],
   "source": [
    "print(results4)"
   ]
  }
 ],
 "metadata": {
  "accelerator": "GPU",
  "colab": {
   "gpuType": "T4",
   "provenance": []
  },
  "kaggle": {
   "accelerator": "nvidiaTeslaT4",
   "dataSources": [],
   "dockerImageVersionId": 31089,
   "isGpuEnabled": true,
   "isInternetEnabled": true,
   "language": "python",
   "sourceType": "notebook"
  },
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
