{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Paramters :  Namespace(rounds=81, num_users=100, nclass=2, nsample_pc=250, frac=0.2, local_ep=10, local_bs=10, bs=128, lr=0.01, momentum=0.5, warmup_epoch=0, trial=1, mu=0.001, model='simple-cnn', ks=5, in_ch=3, dataset='cifar10', noniid=False, shard=False, label=False, split_test=False, savedir='../save_results/', datadir='../data/', logdir='../logs/', partition='flag-non-iid', alg='pacfl', beta=1, local_view=True, batch_size=64, noise=0, noise_type='level', cluster_alpha=1.37, n_basis=3, linkage='average', nclasses=10, nsamples_shared=2500, nclusters=3, num_incluster_layers=2, pruning_percent=10, pruning_target=30, dist_thresh=0.0001, acc_thresh=50, weight_decay=0.0001, gpu=-1, is_print=False, print_freq=10, seed=1, load_initial='', device=device(type='cpu'))\n"
     ]
    }
   ],
   "source": [
    "#packages\n",
    "\n",
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "%reload_ext autoreload\n",
    "\n",
    "\n",
    "import numpy as np\n",
    "\n",
    "import copy\n",
    "import os \n",
    "import gc \n",
    "import pickle\n",
    "import time\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "\n",
    "\n",
    "import torch\n",
    "from torch import nn\n",
    "import torch.nn.functional as F\n",
    "from torch.utils.data import DataLoader, Dataset\n",
    "from torchvision import datasets, transforms\n",
    "\n",
    "from src.data import *\n",
    "from src.models import *\n",
    "from src.fedavg import *\n",
    "from src.client import * \n",
    "from src.clustering import *\n",
    "from src.utils import * \n",
    "\n",
    "\n",
    "from scipy.sparse.csgraph import connected_components\n",
    "from scipy.sparse import csr_matrix\n",
    "\n",
    "\n",
    "st=time.time()\n",
    "args = args_parser()\n",
    "\n",
    "args.device = torch.device('cuda:{}'.format(args.gpu) if torch.cuda.is_available() else 'cpu')\n",
    "\n",
    "torch.cuda.set_device(args.gpu) ## Setting cuda on GPU \n",
    "\n",
    "def mkdirs(dirpath):\n",
    "    try:\n",
    "        os.makedirs(dirpath)\n",
    "    except Exception as _:\n",
    "        pass\n",
    "\n",
    "#parameters for flag setup\n",
    "args.local_view=True\n",
    "args.model='simple-cnn'\n",
    "args.dataset='cifar10'\n",
    "args.partition='flag-non-iid'\n",
    "args.num_users=100\n",
    "args.rounds=81\n",
    "args.frac=.2\n",
    "args.beta=1\n",
    "r=1.5\n",
    "print(\"Paramters : \",str(args))\n",
    "path = args.savedir + args.alg + '/' + args.partition + '/' + args.dataset + '/'\n",
    "mkdirs(path)\n",
    "\n",
    "template = \"Algorithm {}, Clients {}, Dataset {}, Model {}, Non-IID {}, Threshold {}, K {}, Linkage {}, LR {}, Ep {}, Rounds {}, bs {}, frac {}\"\n",
    "\n",
    "s = template.format(args.alg, args.num_users, args.dataset, args.model, args.partition, args.cluster_alpha, args.n_basis, args.linkage, args.lr, args.local_ep, args.rounds, args.local_ep, args.frac)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "partition: flag-non-iid\n",
      "Data statistics Train:\n",
      " {0: {4: 82, 5: 896, 9: 136}, 1: {3: 198, 4: 5, 8: 61}, 2: {3: 89, 4: 31, 8: 47}, 3: {0: 388, 1: 267, 2: 221}, 4: {0: 19, 1: 589, 2: 503}, 5: {3: 2, 4: 42, 8: 304}, 6: {4: 45, 7: 30, 9: 35}, 7: {3: 72, 4: 42, 8: 98}, 8: {3: 214, 6: 93, 8: 37}, 9: {3: 30, 4: 85, 8: 9}, 10: {4: 84, 5: 444, 9: 13}, 11: {4: 82, 5: 39, 9: 61}, 12: {3: 156, 6: 65, 8: 31}, 13: {4: 33, 5: 413, 9: 130}, 14: {4: 66, 7: 4, 9: 68}, 15: {3: 39, 4: 26, 8: 442}, 16: {4: 106, 7: 426, 9: 109}, 17: {3: 120, 4: 111, 8: 68}, 18: {4: 30, 7: 226, 9: 181}, 19: {3: 1, 6: 253, 8: 201}, 20: {3: 95, 6: 315, 8: 17}, 21: {3: 162, 4: 61, 8: 38}, 22: {4: 43, 7: 219, 9: 34}, 23: {4: 51, 5: 161, 9: 52}, 24: {3: 290, 4: 1, 8: 31}, 25: {0: 439, 1: 272, 2: 757}, 26: {4: 130, 7: 87, 9: 127}, 27: {4: 10, 5: 60, 9: 180}, 28: {3: 55, 4: 18, 8: 221}, 29: {3: 12, 6: 460, 8: 13}, 30: {4: 265, 5: 260, 9: 11}, 31: {4: 62, 7: 68, 9: 39}, 32: {0: 1238, 1: 2500, 2: 15}, 33: {4: 71, 7: 284, 9: 125}, 34: {0: 333, 1: 50, 2: 117}, 35: {4: 22, 7: 12, 9: 34}, 36: {3: 36, 6: 108, 8: 40}, 37: {3: 31, 4: 95, 8: 28}, 38: {3: 427, 6: 84, 8: 31}, 39: {4: 17, 5: 195, 9: 8}, 40: {4: 53, 5: 152, 9: 224}, 41: {4: 64, 7: 670, 9: 4}, 42: {0: 559, 1: 241, 2: 1137}, 43: {4: 53, 5: 341, 9: 37}, 44: {3: 245, 4: 42, 8: 52}, 45: {3: 124, 6: 115, 8: 198}, 46: {3: 1, 6: 12, 8: 218}, 47: {3: 30, 4: 109, 8: 328}, 48: {4: 9, 7: 142, 9: 31}, 49: {3: 85, 6: 465, 8: 373}, 50: {0: 210, 1: 195, 2: 357}, 51: {0: 120, 1: 10, 2: 88}, 52: {4: 151, 7: 52, 9: 96}, 53: {0: 586, 1: 147, 2: 214}, 54: {4: 2, 5: 130, 9: 193}, 55: {3: 7, 4: 18, 8: 9}, 56: {4: 15, 5: 86, 9: 322}, 57: {0: 168, 1: 332, 2: 1112}, 58: {4: 106, 7: 144, 9: 19}, 59: {4: 323, 7: 44, 9: 100}, 60: {4: 83, 7: 118, 9: 93}, 61: {3: 43, 4: 312, 8: 130}, 62: {0: 940, 1: 397, 2: 479}, 63: {3: 420, 6: 1370, 8: 276}, 64: {3: 18, 6: 588, 8: 37}, 65: {3: 395, 6: 237, 8: 114}, 66: {3: 59, 4: 77, 8: 7}, 67: {4: 25, 5: 38, 9: 58}, 68: {4: 17, 5: 147, 9: 102}, 69: {4: 128, 5: 722, 9: 50}, 70: {3: 92, 4: 16, 8: 79}, 71: {3: 35, 4: 12, 8: 163}, 72: {3: 281, 6: 12, 8: 67}, 73: {4: 175, 5: 8, 9: 570}, 74: {4: 6, 7: 399, 9: 137}, 75: {4: 46, 7: 214, 9: 31}, 76: {4: 87, 5: 39, 9: 27}, 77: {4: 59, 5: 175, 9: 43}, 78: {4: 125, 7: 648, 9: 259}, 79: {3: 175, 6: 317, 8: 19}, 80: {3: 48, 6: 137, 8: 155}, 81: {3: 283, 6: 23, 8: 144}, 82: {4: 62, 7: 75, 9: 105}, 83: {4: 452, 5: 104, 9: 59}, 84: {4: 82, 7: 13, 9: 26}, 85: {3: 3, 4: 41, 8: 126}, 86: {4: 47, 5: 533, 9: 69}, 87: {3: 52, 4: 91, 8: 220}, 88: {4: 66, 7: 261, 9: 230}, 89: {4: 27, 5: 57, 9: 32}, 90: {3: 126, 6: 67, 8: 225}, 91: {4: 59, 7: 69, 9: 46}, 92: {3: 7, 4: 3, 8: 170}, 93: {4: 78, 7: 50, 9: 289}, 94: {4: 50, 7: 118, 9: 290}, 95: {4: 34, 7: 620, 9: 4}, 96: {3: 10, 6: 258, 8: 119}, 97: {3: 247, 4: 77, 8: 30}, 98: {4: 2, 7: 7, 9: 111}, 99: {3: 185, 6: 21, 8: 24}} \n",
      "\n",
      "Data statistics Test:\n",
      " {0: {4: 1000, 5: 1000, 9: 1000}, 1: {3: 1000, 4: 1000, 8: 1000}, 2: {3: 1000, 4: 1000, 8: 1000}, 3: {0: 1000, 1: 1000, 2: 1000}, 4: {0: 1000, 1: 1000, 2: 1000}, 5: {3: 1000, 4: 1000, 8: 1000}, 6: {4: 1000, 7: 1000, 9: 1000}, 7: {3: 1000, 4: 1000, 8: 1000}, 8: {3: 1000, 6: 1000, 8: 1000}, 9: {3: 1000, 4: 1000, 8: 1000}, 10: {4: 1000, 5: 1000, 9: 1000}, 11: {4: 1000, 5: 1000, 9: 1000}, 12: {3: 1000, 6: 1000, 8: 1000}, 13: {4: 1000, 5: 1000, 9: 1000}, 14: {4: 1000, 7: 1000, 9: 1000}, 15: {3: 1000, 4: 1000, 8: 1000}, 16: {4: 1000, 7: 1000, 9: 1000}, 17: {3: 1000, 4: 1000, 8: 1000}, 18: {4: 1000, 7: 1000, 9: 1000}, 19: {3: 1000, 6: 1000, 8: 1000}, 20: {3: 1000, 6: 1000, 8: 1000}, 21: {3: 1000, 4: 1000, 8: 1000}, 22: {4: 1000, 7: 1000, 9: 1000}, 23: {4: 1000, 5: 1000, 9: 1000}, 24: {3: 1000, 4: 1000, 8: 1000}, 25: {0: 1000, 1: 1000, 2: 1000}, 26: {4: 1000, 7: 1000, 9: 1000}, 27: {4: 1000, 5: 1000, 9: 1000}, 28: {3: 1000, 4: 1000, 8: 1000}, 29: {3: 1000, 6: 1000, 8: 1000}, 30: {4: 1000, 5: 1000, 9: 1000}, 31: {4: 1000, 7: 1000, 9: 1000}, 32: {0: 1000, 1: 1000, 2: 1000}, 33: {4: 1000, 7: 1000, 9: 1000}, 34: {0: 1000, 1: 1000, 2: 1000}, 35: {4: 1000, 7: 1000, 9: 1000}, 36: {3: 1000, 6: 1000, 8: 1000}, 37: {3: 1000, 4: 1000, 8: 1000}, 38: {3: 1000, 6: 1000, 8: 1000}, 39: {4: 1000, 5: 1000, 9: 1000}, 40: {4: 1000, 5: 1000, 9: 1000}, 41: {4: 1000, 7: 1000, 9: 1000}, 42: {0: 1000, 1: 1000, 2: 1000}, 43: {4: 1000, 5: 1000, 9: 1000}, 44: {3: 1000, 4: 1000, 8: 1000}, 45: {3: 1000, 6: 1000, 8: 1000}, 46: {3: 1000, 6: 1000, 8: 1000}, 47: {3: 1000, 4: 1000, 8: 1000}, 48: {4: 1000, 7: 1000, 9: 1000}, 49: {3: 1000, 6: 1000, 8: 1000}, 50: {0: 1000, 1: 1000, 2: 1000}, 51: {0: 1000, 1: 1000, 2: 1000}, 52: {4: 1000, 7: 1000, 9: 1000}, 53: {0: 1000, 1: 1000, 2: 1000}, 54: {4: 1000, 5: 1000, 9: 1000}, 55: {3: 1000, 4: 1000, 8: 1000}, 56: {4: 1000, 5: 1000, 9: 1000}, 57: {0: 1000, 1: 1000, 2: 1000}, 58: {4: 1000, 7: 1000, 9: 1000}, 59: {4: 1000, 7: 1000, 9: 1000}, 60: {4: 1000, 7: 1000, 9: 1000}, 61: {3: 1000, 4: 1000, 8: 1000}, 62: {0: 1000, 1: 1000, 2: 1000}, 63: {3: 1000, 6: 1000, 8: 1000}, 64: {3: 1000, 6: 1000, 8: 1000}, 65: {3: 1000, 6: 1000, 8: 1000}, 66: {3: 1000, 4: 1000, 8: 1000}, 67: {4: 1000, 5: 1000, 9: 1000}, 68: {4: 1000, 5: 1000, 9: 1000}, 69: {4: 1000, 5: 1000, 9: 1000}, 70: {3: 1000, 4: 1000, 8: 1000}, 71: {3: 1000, 4: 1000, 8: 1000}, 72: {3: 1000, 6: 1000, 8: 1000}, 73: {4: 1000, 5: 1000, 9: 1000}, 74: {4: 1000, 7: 1000, 9: 1000}, 75: {4: 1000, 7: 1000, 9: 1000}, 76: {4: 1000, 5: 1000, 9: 1000}, 77: {4: 1000, 5: 1000, 9: 1000}, 78: {4: 1000, 7: 1000, 9: 1000}, 79: {3: 1000, 6: 1000, 8: 1000}, 80: {3: 1000, 6: 1000, 8: 1000}, 81: {3: 1000, 6: 1000, 8: 1000}, 82: {4: 1000, 7: 1000, 9: 1000}, 83: {4: 1000, 5: 1000, 9: 1000}, 84: {4: 1000, 7: 1000, 9: 1000}, 85: {3: 1000, 4: 1000, 8: 1000}, 86: {4: 1000, 5: 1000, 9: 1000}, 87: {3: 1000, 4: 1000, 8: 1000}, 88: {4: 1000, 7: 1000, 9: 1000}, 89: {4: 1000, 5: 1000, 9: 1000}, 90: {3: 1000, 6: 1000, 8: 1000}, 91: {4: 1000, 7: 1000, 9: 1000}, 92: {3: 1000, 4: 1000, 8: 1000}, 93: {4: 1000, 7: 1000, 9: 1000}, 94: {4: 1000, 7: 1000, 9: 1000}, 95: {4: 1000, 7: 1000, 9: 1000}, 96: {3: 1000, 6: 1000, 8: 1000}, 97: {3: 1000, 4: 1000, 8: 1000}, 98: {4: 1000, 7: 1000, 9: 1000}, 99: {3: 1000, 6: 1000, 8: 1000}} \n",
      "\n"
     ]
    }
   ],
   "source": [
    "##################################### Data partitioning section \n",
    "args.local_view = True\n",
    "X_train, y_train, X_test, y_test, net_dataidx_map, net_dataidx_map_test, \\\n",
    "traindata_cls_counts, testdata_cls_counts = partition_data(args.dataset, \n",
    "args.datadir, args.logdir, args.partition, args.num_users, beta=args.beta, local_view=args.local_view)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "len train_ds_global: 50000\n",
      "len test_ds_global: 10000\n",
      "MODEL: simple-cnn, Dataset: cifar10\n",
      "SimpleCNN(\n",
      "  (encoder): Sequential(\n",
      "    (0): Conv2d(3, 64, kernel_size=(3, 3), stride=(1, 1))\n",
      "    (1): ReLU()\n",
      "    (2): MaxPool2d(kernel_size=2, stride=2, padding=0, dilation=1, ceil_mode=False)\n",
      "    (3): Conv2d(64, 128, kernel_size=(3, 3), stride=(1, 1))\n",
      "    (4): ReLU()\n",
      "    (5): MaxPool2d(kernel_size=2, stride=2, padding=0, dilation=1, ceil_mode=False)\n",
      "    (6): Conv2d(128, 256, kernel_size=(3, 3), stride=(1, 1))\n",
      "    (7): ReLU()\n",
      "    (8): MaxPool2d(kernel_size=2, stride=2, padding=0, dilation=1, ceil_mode=False)\n",
      "    (9): Flatten(start_dim=1, end_dim=-1)\n",
      "    (10): Dropout(p=0.5, inplace=False)\n",
      "    (11): Linear(in_features=1024, out_features=128, bias=True)\n",
      "    (12): ReLU()\n",
      "    (13): Dropout(p=0.5, inplace=False)\n",
      "    (14): Linear(in_features=128, out_features=256, bias=True)\n",
      "    (15): ReLU()\n",
      "  )\n",
      "  (classifier): Linear(in_features=256, out_features=10, bias=True)\n",
      ")\n",
      "encoder.0.weight torch.Size([64, 3, 3, 3])\n",
      "encoder.0.bias torch.Size([64])\n",
      "encoder.3.weight torch.Size([128, 64, 3, 3])\n",
      "encoder.3.bias torch.Size([128])\n",
      "encoder.6.weight torch.Size([256, 128, 3, 3])\n",
      "encoder.6.bias torch.Size([256])\n",
      "encoder.11.weight torch.Size([128, 1024])\n",
      "encoder.11.bias torch.Size([128])\n",
      "encoder.14.weight torch.Size([256, 128])\n",
      "encoder.14.bias torch.Size([256])\n",
      "classifier.weight torch.Size([10, 256])\n",
      "classifier.bias torch.Size([10])\n",
      "537610\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 5 9], Counts: [ 82 896 136]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [198   5  61]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [89 31 47]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [0 1 2], Counts: [388 267 221]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [0 1 2], Counts: [ 19 589 503]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [  2  42 304]\n",
      "Shape of U: (3072, 8)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [45 30 35]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [72 42 98]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 6 8], Counts: [214  93  37]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [30 85  9]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 5 9], Counts: [ 84 444  13]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 5 9], Counts: [82 39 61]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 6 8], Counts: [156  65  31]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 5 9], Counts: [ 33 413 130]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [66  4 68]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [ 39  26 442]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [106 426 109]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [120 111  68]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [ 30 226 181]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 6 8], Counts: [  1 253 201]\n",
      "Shape of U: (3072, 7)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 6 8], Counts: [ 95 315  17]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [162  61  38]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [ 43 219  34]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 5 9], Counts: [ 51 161  52]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [290   1  31]\n",
      "Shape of U: (3072, 7)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [0 1 2], Counts: [439 272 757]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [130  87 127]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 5 9], Counts: [ 10  60 180]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [ 55  18 221]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 6 8], Counts: [ 12 460  13]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 5 9], Counts: [265 260  11]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [62 68 39]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [0 1 2], Counts: [1238 2500   15]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [ 71 284 125]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [0 1 2], Counts: [333  50 117]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [22 12 34]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 6 8], Counts: [ 36 108  40]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [31 95 28]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 6 8], Counts: [427  84  31]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 5 9], Counts: [ 17 195   8]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 5 9], Counts: [ 53 152 224]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [ 64 670   4]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [0 1 2], Counts: [ 559  241 1137]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 5 9], Counts: [ 53 341  37]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [245  42  52]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 6 8], Counts: [124 115 198]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 6 8], Counts: [  1  12 218]\n",
      "Shape of U: (3072, 7)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [ 30 109 328]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [  9 142  31]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 6 8], Counts: [ 85 465 373]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [0 1 2], Counts: [210 195 357]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [0 1 2], Counts: [120  10  88]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [151  52  96]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [0 1 2], Counts: [586 147 214]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 5 9], Counts: [  2 130 193]\n",
      "Shape of U: (3072, 8)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [ 7 18  9]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 5 9], Counts: [ 15  86 322]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [0 1 2], Counts: [ 168  332 1112]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [106 144  19]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [323  44 100]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [4 7 9], Counts: [ 83 118  93]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 4 8], Counts: [ 43 312 130]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [0 1 2], Counts: [940 397 479]\n",
      "Shape of U: (3072, 9)\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Labels: [3 6 8], Counts: [ 420 1370  276]\n"
     ]
    }
   ],
   "source": [
    "train_dl_global, test_dl_global, train_ds_global, test_ds_global = get_dataloader(args.dataset,\n",
    "                                                                                   args.datadir,\n",
    "                                                                                   args.batch_size,\n",
    "                                                                                   32)\n",
    "\n",
    "print(\"len train_ds_global:\", len(train_ds_global))\n",
    "print(\"len test_ds_global:\", len(test_ds_global))\n",
    "\n",
    "\n",
    "################################### build  single fe+cl model\n",
    "def init_nets(args, dropout_p=0.5):\n",
    "\n",
    "    users_model = []\n",
    "\n",
    "    for net_i in range(-1, args.num_users):\n",
    "        if args.model == \"simple-cnn\":\n",
    "            if args.dataset in (\"cifar10\", \"cinic10\", \"svhn\"):\n",
    "                #net = CNN_CIFAR.to(args.device) #changed delete\n",
    "                # net = SimpleCNNLight(input_dim=(16 * 5 * 5), hidden_dims=[120, 84], output_dim=10).to(args.device)\n",
    "                net = SimpleCNN(num_classes=10).to(args.device) \n",
    "            elif args.dataset in (\"mnist\", 'femnist', 'fmnist'):\n",
    "                net = SimpleCNNMNIST2().to(args.device)\n",
    "               # net = SimpleCNNMNIST(input_dim=(16 * 4 * 4), hidden_dims=[120, 84], output_dim=10).to(args.device)\n",
    "            elif args.dataset == 'cifar100':\n",
    "                #net = SimpleCNNLight(input_dim=(16 * 5 * 5), hidden_dims=[120, 84], output_dim=100).to(args.device)\n",
    " \n",
    "                net = SimpleCNN_3(input_dim=(16 * 3 * 5 * 5), hidden_dims=[120*3, 84*3], output_dim=100).to(args.device)\n",
    "                #\n",
    "        elif args.model ==\"simple-cnn-3\":\n",
    "            if args.dataset == 'cifar100': \n",
    "                net = SimpleCNN_3(input_dim=(16 * 3 * 5 * 5), hidden_dims=[120*3, 84*3], output_dim=100).to(args.device)\n",
    "            if args.dataset == 'tinyimagenet':\n",
    "                net = SimpleCNNTinyImagenet_3(input_dim=(16 * 3 * 13 * 13), hidden_dims=[120*3, 84*3], \n",
    "                                              output_dim=200).to(args.device)\n",
    "        # elif args.model == \"vgg-9\":\n",
    "        #     if args.dataset in (\"mnist\", 'femnist'):\n",
    "        #         net = ModerateCNNMNIST().to(args.device)\n",
    "        #     elif args.dataset in (\"cifar10\", \"cinic10\", \"svhn\"):\n",
    "        #         # print(\"in moderate cnn\")\n",
    "        #         net = ModerateCNN().to(args.device)\n",
    "        #     elif args.dataset == 'celeba':\n",
    "        #         net = ModerateCNN(output_dim=2).to(args.device)\n",
    "        elif args.model == 'resnet9': \n",
    "            if args.dataset == 'cifar100':\n",
    "                net = SimpleCNN_3(input_dim=(16 * 3 * 5 * 5), hidden_dims=[120*3, 84*3], output_dim=100).to(args.device)\n",
    "                #net = ResNet9(in_channels=3, num_classes=100)\n",
    "            elif args.dataset == 'tinyimagenet': \n",
    "                net = ResNet9(in_channels=3, num_classes=200, dim=512*2*2)\n",
    "        else:\n",
    "            print(\"not supported yet\")\n",
    "            exit(1)\n",
    "        if net_i == -1: #initializing the global model\n",
    "            net_glob = copy.deepcopy(net) \n",
    "           \n",
    "            initial_state_dict = copy.deepcopy(net_glob.state_dict())\n",
    "            server_state_dict = copy.deepcopy(net_glob.state_dict())\n",
    "            if args.load_initial:\n",
    "                initial_state_dict = torch.load(args.load_initial)\n",
    "                server_state_dict = torch.load(args.load_initial)\n",
    "                net_glob.load_state_dict(initial_state_dict)\n",
    " \n",
    "        else:\n",
    "            users_model.append(copy.deepcopy(net))\n",
    "            users_model[net_i].load_state_dict(initial_state_dict)\n",
    "\n",
    "#     model_meta_data = []\n",
    "#     layer_type = []\n",
    "#     for (k, v) in nets[0].state_dict().items():\n",
    "#         model_meta_data.append(v.shape)\n",
    "#         layer_type.append(k)\n",
    "\n",
    "    return users_model, net_glob, initial_state_dict, server_state_dict\n",
    "\n",
    "print(f'MODEL: {args.model}, Dataset: {args.dataset}')\n",
    "\n",
    "users_model, net_glob, initial_state_dict, server_state_dict = init_nets(args, dropout_p=0.5)\n",
    "\n",
    "print(net_glob)\n",
    "\n",
    "total = 0 \n",
    "for name, param in net_glob.named_parameters():\n",
    "    print(name, param.size())\n",
    "    total += np.prod(param.size())\n",
    "    #print(np.array(param.data.cpu().numpy().reshape([-1])))\n",
    "    #print(isinstance(param.data.cpu().numpy(), np.array))\n",
    "print(total)\n",
    "\n",
    "################################# Initializing Clients \n",
    "traindata_cls_ratio = {}\n",
    "pretrain_data1=[] #changed\n",
    "pretrain_data2=[] #changed\n",
    "\n",
    "budget = 50\n",
    "for i in range(args.num_users):\n",
    "    total_sum = sum(list(traindata_cls_counts[i].values()))\n",
    "    base = 1/len(list(traindata_cls_counts[i].values()))\n",
    "    temp_ratio = {}\n",
    "    for k in traindata_cls_counts[i].keys():\n",
    "        ss = traindata_cls_counts[i][k]/total_sum\n",
    "        temp_ratio[k] = (traindata_cls_counts[i][k]/total_sum)\n",
    "        # if ss >= (base + 0.05): \n",
    "        #     temp_ratio[k] = traindata_cls_counts[i][k]  #if any ratio excceds the base, use the \n",
    "            #actual value instead of ratios\n",
    "            \n",
    "    sub_sum = sum(list(temp_ratio.values()))\n",
    "    for k in temp_ratio.keys():\n",
    "        temp_ratio[k] = (temp_ratio[k]/sub_sum)*budget      #normalize the ratios to budget\n",
    "    \n",
    "    round_ratio = round_to(list(temp_ratio.values()), budget) #round ratios so they are int\n",
    "    cnt = 0 \n",
    "    for k in temp_ratio.keys():\n",
    "        temp_ratio[k] = round_ratio[cnt]\n",
    "        cnt+=1\n",
    "        \n",
    "    traindata_cls_ratio[i] = temp_ratio  \n",
    "    \n",
    "clients = []\n",
    "U_clients = []\n",
    "\n",
    "K = args.n_basis\n",
    "#K = 5\n",
    "U_dict={}\n",
    "for idx in range(args.num_users):\n",
    "    \n",
    "    dataidxs = net_dataidx_map[idx]\n",
    "    if net_dataidx_map_test is None:\n",
    "        dataidx_test = None \n",
    "    else:\n",
    "        dataidxs_test = net_dataidx_map_test[idx]\n",
    "\n",
    "    #print(f'Initializing Client {idx}')\n",
    "\n",
    "    noise_level = args.noise\n",
    "    if idx == args.num_users - 1:\n",
    "        noise_level = 0\n",
    "\n",
    "    if args.noise_type == 'space':\n",
    "        train_dl_local, test_dl_local, train_ds_local, test_ds_local, pre_train_dl, validation_dl = get_dataloader(args.dataset, \n",
    "                                                                       args.datadir, args.local_bs, 32, \n",
    "                                                                       dataidxs, noise_level, idx, \n",
    "                                                                       args.num_users-1, \n",
    "                                                                       dataidxs_test=dataidxs_test)\n",
    "    else:\n",
    "        noise_level = args.noise / (args.num_users - 1) * idx\n",
    "        train_dl_local, test_dl_local, train_ds_local, test_ds_local, pre_train_dl, validation_dl = get_dataloader(args.dataset, \n",
    "                                                                       args.datadir, args.local_bs, 32, \n",
    "                                                                       dataidxs, noise_level, \n",
    "                                                                       dataidxs_test=dataidxs_test)\n",
    "    idxs_local = np.arange(len(train_ds_local.data))\n",
    "    labels_local = np.array(train_ds_local.target)\n",
    "    # Sort Labels Train \n",
    "    idxs_labels_local = np.vstack((idxs_local, labels_local))\n",
    "    idxs_labels_local = idxs_labels_local[:, idxs_labels_local[1, :].argsort()]\n",
    "    idxs_local = idxs_labels_local[0, :]\n",
    "    labels_local = idxs_labels_local[1, :]\n",
    "    \n",
    "    uni_labels, cnt_labels = np.unique(labels_local, return_counts=True)\n",
    "    \n",
    "    print(f'Labels: {uni_labels}, Counts: {cnt_labels}')\n",
    "    \n",
    "    nlabels = len(uni_labels)\n",
    "    cnt = 0\n",
    "    U_temp = []\n",
    "    D_temp = {}\n",
    "    for j in range(nlabels):\n",
    "        local_ds1 = train_ds_local.data[idxs_local[cnt:cnt+cnt_labels[j]]]\n",
    "        x=local_ds1.shape[0]\n",
    "        local_ds1_copy = copy.deepcopy(local_ds1)\n",
    "        #print(train_ds_local[idxs_local[cnt]][0])\n",
    "        #show_tensor_image(train_ds_local[idxs_local[cnt]][0])\n",
    "        local_ds1 = local_ds1.reshape(cnt_labels[j], -1)\n",
    "         \n",
    "        local_ds1 = local_ds1.T\n",
    "        if type(train_ds_local.target[idxs_local[cnt:cnt+cnt_labels[j]]]) == torch.Tensor:\n",
    "            label1 = list(set(train_ds_local.target[idxs_local[cnt:cnt+cnt_labels[j]]].numpy()))\n",
    "        else:\n",
    "            label1 = list(set(train_ds_local.target[idxs_local[cnt:cnt+cnt_labels[j]]]))\n",
    "        assert len(label1) == 1\n",
    "        \n",
    "        #print(f'Label {j} : {label1}')\n",
    "        \n",
    "        \n",
    "        #pretrain_data1.extend(select_representative_images(copy.deepcopy(local_ds1_copy),cnt_labels[j], 8, 30, label1[0])) #changed, delete, para=2nd\n",
    "        # if local_ds1_copy.shape[0]>1:\n",
    "        #     pretrain_data2.extend(select_representative_images_with_svd(copy.deepcopy(local_ds1_copy), x, label1[0]))\n",
    "        if args.partition == 'noniid-labeldir': \n",
    "            #print('Dir partition')\n",
    "            if label1 in list(traindata_cls_ratio[idx].keys()): \n",
    "                K = traindata_cls_ratio[idx][label1[0]]\n",
    "            else: \n",
    "                K = min(args.n_basis,x)    #changed, K = args.n_basis\n",
    "        if K > 0:\n",
    "            u1_temp, sh1_temp, vh1_temp = np.linalg.svd(local_ds1, full_matrices=False)\n",
    "            u1_temp=u1_temp/np.linalg.norm(u1_temp, ord=2, axis=0)\n",
    "            U_temp.append(u1_temp[:, 0:K])\n",
    "            D_temp[label1[0]] = u1_temp[:, 0:K] #changed\n",
    "            \n",
    "        cnt+=cnt_labels[j]\n",
    "        \n",
    "    #U_temp = [u1_temp[:, 0:K], u2_temp[:, 0:K]]\n",
    "    U_clients.append(copy.deepcopy(np.hstack(U_temp)))\n",
    "    U_dict[idx]= D_temp\n",
    "    \n",
    "    print(f'Shape of U: {U_clients[-1].shape}')\n",
    "    \n",
    "    clients.append(Client_ClusterFL(idx, copy.deepcopy(users_model[idx]), args.local_bs, args.local_ep, \n",
    "               args.lr, args.momentum, args.device, train_dl_local, test_dl_local,  pre_train_dl, validation_dl))\n",
    "\n",
    "############### Getting gradient similarity matrix\n",
    "clients_backup = [copy.deepcopy(client) for client in clients] # Make a backup of clients' models\n",
    "\n",
    "# --- Added: build mask using dimension of one flattened gradient\n",
    "#    Used to pick random indices for all clients graidients, a small subset for compression\n",
    "#    Assosscation: compress_gradient_with_mask, generate_sparsity_mask, Modified pairwise_angles\n",
    "#    Assosscation: Modified: flatten + sparsify in gradient similarity\n",
    "compression_ratio = 50            # e.g. keep 1/100 of coordinates\n",
    "first_diff = clients[0].get_W()\n",
    "first_flat = flatten(first_diff).detach().cpu().numpy()\n",
    "D = first_flat.shape[0]\n",
    "mask = generate_sparsity_mask(D, compression_ratio)\n",
    "# --- End of Added\n",
    "\n",
    "list_of_dW=[]\n",
    "#w_glob = copy.deepcopy(initial_state_dict)\n",
    "for iteration in range(1): #run the model for 25 iterations before gettin gradients edited\n",
    "    idxs_users = np.arange(args.num_users) #get all users id\n",
    "    print(f'###### Gradient data ROUND {iteration+1} ######')\n",
    "    list_of_dW.clear()\n",
    "    #get global state and train\n",
    "    for idx in idxs_users:\n",
    "        #clients[idx].set_state_dict(copy.deepcopy(w_glob))\n",
    "        oldW = clients[idx].get_W()\n",
    "        loss = clients[idx].train(is_print=False)\n",
    "        newW = clients[idx].get_W()\n",
    "        difference_W = copy.deepcopy(oldW)\n",
    "        subtract_(target=difference_W, minuend=newW, subtrahend=oldW)\n",
    "        \n",
    "        # --- Modified: flatten + sparsify\n",
    "        flat = flatten(difference_W).detach().cpu().numpy()\n",
    "        sparse = compress_gradient_with_mask(flat, mask)\n",
    "        list_of_dW.append(sparse)\n",
    "        # --- End of Modified\n",
    "        \n",
    "        # list_of_dW.append(difference_W) \n",
    "        #-uncomment incase remove the  \"Modified: flatten + sparsify\",\n",
    "        #i.e, remove gradient compression\n",
    "        \n",
    "       \n",
    "    \n",
    "Grad_similarites = pairwise_angles(list_of_dW) \n",
    "clients = [copy.deepcopy(client) for client in clients_backup] # Reset clients using backup\n",
    "\n",
    "######## Getting class-wise weighted data similarity matrix\n",
    "\n",
    "def compute_w(fc1, fc2):\n",
    "    if fc1 == 0 or fc2 == 0:\n",
    "        return None  # Skip positions where fc1 or fc2 is 0\n",
    "    \n",
    "    max_val = max(fc1, fc2)\n",
    "    min_val = min(fc1, fc2)\n",
    "    \n",
    "    w = max_val/min_val\n",
    "    return w\n",
    "\n",
    "def normalize_weights(weights, r):\n",
    "    # Flatten the weights matrix while excluding None values\n",
    "    valid_weights = [w for w in weights.flatten() if w is not None]\n",
    "    \n",
    "    min_weight = min(valid_weights)\n",
    "    max_weight = max(valid_weights)\n",
    "    \n",
    "    # Normalize weights to the range [1, 1 + r]\n",
    "    normalized_weights = np.full_like(weights, None, dtype=np.float64)\n",
    "    for idx in np.ndindex(weights.shape):\n",
    "        if weights[idx] is not None:\n",
    "            normalized_weights[idx] = (weights[idx] - min_weight) / (max_weight - min_weight) * r + 1\n",
    "   \n",
    "    return normalized_weights\n",
    "\n",
    "def calculating_weighted_adjacency(nclients, U_dict, labels, traindata_cls_counts, r):\n",
    "    sim_mat = np.zeros([nclients, nclients])\n",
    "    weights = np.full([nclients, nclients, labels], None)  # 3D array to store weights, initialized with None\n",
    "    \n",
    "    # Compute and store weights\n",
    "    for idx1 in range(nclients):\n",
    "        for idx2 in range(nclients):\n",
    "            if idx1 == idx2:\n",
    "                continue\n",
    "            for l in range(labels):\n",
    "                if l in traindata_cls_counts[idx1] and l in traindata_cls_counts[idx2]:\n",
    "                    f1 = traindata_cls_counts[idx1][l]\n",
    "                    f2 = traindata_cls_counts[idx2][l]\n",
    "                    weights[idx1, idx2, l] = compute_w(f1, f2)  # Parameter b\n",
    "    \n",
    "    # Normalize the weights matrix\n",
    "    normalized_weights = normalize_weights(weights, r)\n",
    "    \n",
    "    # Compute the similarity matrix using normalized weights\n",
    "    for idx1 in range(nclients):\n",
    "        for idx2 in range(nclients):\n",
    "            if idx1 == idx2: \n",
    "                sim_mat[idx1, idx2] = 0\n",
    "                continue\n",
    "            \n",
    "            angles = []\n",
    "            for l in range(labels):\n",
    "                if l in U_dict[idx1] and l in U_dict[idx2]:\n",
    "                    U1 = copy.deepcopy(U_dict[idx1][l])\n",
    "                    U2 = copy.deepcopy(U_dict[idx2][l])\n",
    "                    mul = np.clip(U1.T @ U2, a_min=-1.0, a_max=1.0)\n",
    "                    angle = np.min(np.arccos(mul)) * 180 / np.pi\n",
    "                    weight = normalized_weights[idx1, idx2, l]\n",
    "                    if weight is not None:\n",
    "                        angle = min(angle * weight, 180)\n",
    "                    angles.append(angle)\n",
    "                elif l not in traindata_cls_counts[idx1] and l not in traindata_cls_counts[idx2]:\n",
    "                    angles.append(0)\n",
    "                else:\n",
    "                    angles.append(180)\n",
    "            \n",
    "            sim_mat[idx1, idx2] = sum(angles) / len(angles)\n",
    "    \n",
    "    return sim_mat\n",
    "\n",
    "v=calculating_weighted_adjacency(args.num_users, U_dict, 10,traindata_cls_counts,.5) #label hardcode\n",
    "G= normalize_matrix(Grad_similarites)\n",
    "v= normalize_matrix(v)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Learned row-gates w (first 10 shown): [0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000\n",
      " 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 1.0000\n",
      " 0.0000 0.0000 0.0000 1.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000\n",
      " 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 1.0000 0.0000 0.0000 0.0000\n",
      " 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0003 0.0000\n",
      " 0.0000 0.0000 0.0000 0.0000 1.0000 0.0000 0.0000 0.0000 0.0000 0.0000\n",
      " 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000\n",
      " 0.0000 0.0000 1.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000\n",
      " 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 1.0000 0.0000 0.0000]\n"
     ]
    }
   ],
   "source": [
    "#Fusion-gate: combing data and gradient\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "\n",
    "# ----------------------------------------------------------------------\n",
    "# 1.  MLP that outputs an n-dimensional sigmoid-gated vector w ∈ (0,1)^n\n",
    "# ----------------------------------------------------------------------\n",
    "class RowGateMLP(nn.Module):\n",
    "    def __init__(self, n: int, hidden: int = 128):\n",
    "        \"\"\"\n",
    "        Args:\n",
    "            n (int)     : number of clients  (size of similarity matrices)\n",
    "            hidden (int): hidden layer width\n",
    "        \"\"\"\n",
    "        super().__init__()\n",
    "        d_in = 2 * n * n            # vec(G) ‖ vec(P)\n",
    "        self.net = nn.Sequential(\n",
    "            nn.Linear(d_in, hidden),\n",
    "            nn.ReLU(inplace=True),\n",
    "            nn.Linear(hidden, n),    # logits for each client\n",
    "            nn.Sigmoid()             # element-wise → w ∈ (0,1)^n\n",
    "        )\n",
    "\n",
    "    def forward(self, G: torch.Tensor, P: torch.Tensor) -> torch.Tensor:\n",
    "        flat = torch.cat([G.flatten(), P.flatten()], dim=0)   # (2n²,)\n",
    "        return self.net(flat)                                 # (n,)\n",
    "\n",
    "# ----------------------------------------------------------------------\n",
    "# 2.  Fusion  (upper triangle uses row-gate, mirrored for symmetry)\n",
    "# ----------------------------------------------------------------------\n",
    "def fuse_similarity(G: torch.Tensor,\n",
    "                    P: torch.Tensor,\n",
    "                    gate_net: RowGateMLP):\n",
    "    \"\"\"\n",
    "    Returns\n",
    "        A : symmetric fused similarity  (n×n)\n",
    "        w : vector of row gates         (n,)\n",
    "    Formula (for i<j):\n",
    "        A_ij = w_i G_ij + (1-w_i) P_ij ;  A_ji = A_ij ;  A_ii = 0\n",
    "    \"\"\"\n",
    "    w = gate_net(G, P)                       # (n,)\n",
    "    n = w.size(0)\n",
    "    w_col = w.view(n, 1)                     # (n,1) for broadcasting\n",
    "\n",
    "    mix = w_col * G + (1 - w_col) * P        # apply gate row-wise\n",
    "    upper = torch.triu(mix, diagonal=1)      # keep strict upper triangle\n",
    "    A = upper + upper.T                      # mirror → symmetric, zero diag\n",
    "\n",
    "    return A, w\n",
    "\n",
    "# ----------------------------------------------------------------------\n",
    "# 3.  Row-softmax entropy loss  (no Laplacian term)\n",
    "# ----------------------------------------------------------------------\n",
    "def entropy_loss(A: torch.Tensor) -> torch.Tensor:\n",
    "    n = A.size(0)\n",
    "    soft = F.softmax(A, dim=1)\n",
    "    logt = torch.clamp(soft, 1e-12, 1).log()\n",
    "    return -(soft * logt).sum() / n          # minimise → sharper rows\n",
    "\n",
    "# ----------------------------------------------------------------------\n",
    "# 4.  Training example  (G and v are provided 100×100 NumPy or Torch)\n",
    "# ----------------------------------------------------------------------\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "\n",
    "def to_tensor(mat):\n",
    "    if torch.is_tensor(mat):\n",
    "        return mat.clone().to(device)        # duplicate to avoid in-place edits\n",
    "    return torch.as_tensor(mat, dtype=torch.float32, device=device)\n",
    "\n",
    "G_t = to_tensor(G)          # gradient similarity (100×100)\n",
    "V_t = to_tensor(v)          # data      similarity (100×100)\n",
    "\n",
    "n = G_t.size(0)\n",
    "gate_net = RowGateMLP(n).to(device)\n",
    "optimizer = torch.optim.Adam(gate_net.parameters(), lr=1e-3)\n",
    "\n",
    "for step in range(50):\n",
    "    optimizer.zero_grad()\n",
    "    A, w = fuse_similarity(G_t, V_t, gate_net)\n",
    "    loss = entropy_loss(A)\n",
    "    loss.backward()\n",
    "    optimizer.step()\n",
    "\n",
    "print(\"Learned row-gates w (first 10 shown):\", w[:90].cpu().detach().numpy())\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Round 1.5\n",
      "Breaking HC\n",
      "\n",
      "Adjacency Matrix\n",
      "[[0.0000 1.0000 1.0000 ... 1.0000 0.0051 1.0000]\n",
      " [1.0000 0.0000 0.3395 ... 1.0000 1.0000 0.6675]\n",
      " [1.0000 0.3395 0.0000 ... 1.0000 1.0000 0.6697]\n",
      " ...\n",
      " [1.0000 1.0000 1.0000 ... 0.0000 1.0000 0.6735]\n",
      " [0.0051 1.0000 1.0000 ... 1.0000 0.0000 1.0000]\n",
      " [1.0000 0.6675 0.6697 ... 0.6735 1.0000 0.0000]]\n",
      "\n",
      "Clusters: \n",
      "[[0, 82, 7, 15, 59, 17, 46, 85, 65, 95, 77, 30, 13, 40, 98, 26, 42, 67, 31, 28, 5, 80, 70], [1, 9, 22, 47, 96, 16, 24, 52, 86, 11, 41, 76, 33, 74, 73, 29, 38, 72, 2, 10, 43, 49, 75, 81, 63, 90, 18, 83, 39, 84, 34, 32, 93, 64, 19, 23, 3, 6, 12, 88, 14, 48, 94, 20, 97, 25, 57, 35, 56, 71, 91, 51, 55, 58, 37, 78, 21, 89, 4, 27, 79, 87, 44, 45, 99, 53, 60, 68, 61, 62, 66, 50, 69, 92, 8, 36, 54]]\n",
      "\n",
      "Number of Clusters 2\n",
      "\n",
      "Cluster 0: 23 \n",
      "Cluster 1: 77 \n",
      "Clients: Cluster_ID \n",
      "{0: 0, 1: 1, 2: 1, 3: 1, 4: 1, 5: 0, 6: 1, 7: 0, 8: 1, 9: 1, 10: 1, 11: 1, 12: 1, 13: 0, 14: 1, 15: 0, 16: 1, 17: 0, 18: 1, 19: 1, 20: 1, 21: 1, 22: 1, 23: 1, 24: 1, 25: 1, 26: 0, 27: 1, 28: 0, 29: 1, 30: 0, 31: 0, 32: 1, 33: 1, 34: 1, 35: 1, 36: 1, 37: 1, 38: 1, 39: 1, 40: 0, 41: 1, 42: 0, 43: 1, 44: 1, 45: 1, 46: 0, 47: 1, 48: 1, 49: 1, 50: 1, 51: 1, 52: 1, 53: 1, 54: 1, 55: 1, 56: 1, 57: 1, 58: 1, 59: 0, 60: 1, 61: 1, 62: 1, 63: 1, 64: 1, 65: 0, 66: 1, 67: 0, 68: 1, 69: 1, 70: 0, 71: 1, 72: 1, 73: 1, 74: 1, 75: 1, 76: 1, 77: 0, 78: 1, 79: 1, 80: 0, 81: 1, 82: 0, 83: 1, 84: 1, 85: 0, 86: 1, 87: 1, 88: 1, 89: 1, 90: 1, 91: 1, 92: 1, 93: 1, 94: 1, 95: 0, 96: 1, 97: 1, 98: 0, 99: 1}\n",
      "Cluster 0, First Client 0: {0: 58, 2: 526, 8: 430}\n",
      "Cluster 1, First Client 1: {1: 78, 3: 237, 4: 43}\n"
     ]
    }
   ],
   "source": [
    "#HR clustering \n",
    "\n",
    "\n",
    "import numpy as np\n",
    "import copy\n",
    "from scipy.sparse.csgraph import connected_components\n",
    "from scipy.sparse import csr_matrix\n",
    "\n",
    "\n",
    "args.cluster_alpha=.86\n",
    "#1.9\n",
    "args.linkage = 'average' \n",
    "np.set_printoptions(precision=4)\n",
    "\n",
    "cnt = args.num_users\n",
    "\n",
    "print(f'Round {r}')\n",
    "clients_idxs = np.arange(cnt)\n",
    "\n",
    "adj_mat=A.detach().cpu().numpy()\n",
    "#adj_mat=  normalize_matrix(v)*(1-grad_co) + grad_co*normalize_matrix((copy.deepcopy(Grad_similarites)))\n",
    "#adj_mat=  normalize_matrix((copy.deepcopy(Grad_similarites)))\n",
    "#adj_mat=  normalize_matrix(v)\n",
    "\n",
    "clusters = hierarchical_clustering(copy.deepcopy(adj_mat), thresh=args.cluster_alpha, linkage='average')\n",
    "\n",
    "cnt+= 10\n",
    "print('')\n",
    "print('Adjacency Matrix')\n",
    "print(adj_mat)\n",
    "print('')\n",
    "print('Clusters: ')\n",
    "print(clusters)\n",
    "print('')\n",
    "print(f'Number of Clusters {len(clusters)}')\n",
    "print('')\n",
    "for jj in range(len(clusters)):\n",
    "    print(f'Cluster {jj}: {len(clusters[jj])} ')\n",
    "    \n",
    "\n",
    "clients_clust_id = {i:None for i in range(args.num_users)}\n",
    "for i in range(args.num_users):\n",
    "    for j in range(len(clusters)):\n",
    "        if i in clusters[j]:\n",
    "            clients_clust_id[i] = j\n",
    "           # print(i, \" Client in\", \"Cluster \", j, traindata_cls_counts[i] )\n",
    "            break\n",
    "print(f'Clients: Cluster_ID \\n{clients_clust_id}')\n",
    "for k in range(len(clusters)):\n",
    "    print(f\"Cluster {k}, First Client {clusters[k][0]}:\", traindata_cls_counts[clusters[k][0]])\n",
    "\n",
    "\n",
    "\n",
    "import numpy as np\n",
    "from typing import Dict, List\n",
    "\n",
    "def build_similarity_graph_rank_supply_nonzero(\n",
    "        traindata_cls_counts: Dict[int, Dict[int, int]],\n",
    "        clusters: List[List[int]],\n",
    "        num_classes: int = 10,\n",
    "        top_k: int = 2\n",
    ") -> Dict[int, List[int]]:\n",
    "    \"\"\"\n",
    "    Build a directed Cluster-Complementarity Graph.\n",
    "    Demand is high for rare classes; supply is high for abundant classes.\n",
    "\n",
    "    Parameters\n",
    "    ----------\n",
    "    traindata_cls_counts : {client_id: {class: count}}\n",
    "    clusters             : list of clusters (each is a list of client IDs)\n",
    "    num_classes          : total number of classes\n",
    "    top_k                : keep this many outgoing edges per cluster\n",
    "\n",
    "    Returns\n",
    "    -------\n",
    "    similarity_graph : {cluster_id: [target clusters]}\n",
    "    \"\"\"\n",
    "    C, K = len(clusters), num_classes\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 1. Per-client rank over *non-zero* classes\n",
    "    # ----------------------------------------------------------\n",
    "    client_rank = [{} for _ in range(len(traindata_cls_counts))]\n",
    "    for cid, counts in traindata_cls_counts.items():\n",
    "        present = [(cls, cnt) for cls, cnt in counts.items() if cnt > 0]\n",
    "        if not present:\n",
    "            continue                                # client has no data\n",
    "        present.sort(key=lambda x: x[1])            # ascending (rarest first)\n",
    "        for r, (cls, _) in enumerate(present):\n",
    "            client_rank[cid][cls] = r               # rank 0 … m_i-1\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 2. Demand weight  w[p,k]   (large if class k is rare in cluster p)\n",
    "    # ----------------------------------------------------------\n",
    "    w = np.zeros((C, K), dtype=float)\n",
    "    for p, clist in enumerate(clusters):\n",
    "        for cid in clist:\n",
    "            m_i = len(client_rank[cid])             # number of classes present\n",
    "            for cls, r in client_rank[cid].items():\n",
    "                w[p, cls] += (m_i - r)              # larger when rarer\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 3. Supply strength s[q,k] (large if class k is abundant in cluster q)\n",
    "    # ----------------------------------------------------------\n",
    "    s = np.zeros((C, K), dtype=float)\n",
    "    for q, clist in enumerate(clusters):\n",
    "        for cid in clist:\n",
    "            for cls, r in client_rank[cid].items():\n",
    "                s[q, cls] += (r + 1)                # larger when more common\n",
    "        n_q = max(len(clist), 1)\n",
    "        s[q] /= n_q                                 # per-client normalisation\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 4. Complementarity score matrix  A = W · Sᵀ\n",
    "    # ----------------------------------------------------------\n",
    "    score = w @ s.T\n",
    "    np.fill_diagonal(score, -np.inf)                # forbid self-loops\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 5. Keep top-k outgoing edges for each cluster\n",
    "    # ----------------------------------------------------------\n",
    "    graph = {p: np.argsort(-score[p])[:top_k].tolist() for p in range(C)}\n",
    "    return graph\n",
    "similarity_graph = build_similarity_graph_rank_supply_nonzero(\n",
    "    traindata_cls_counts,\n",
    "    clusters,\n",
    "    num_classes=10,\n",
    "    top_k=2\n",
    ")\n",
    "\n",
    "# convert “out-list” format {p:[q1,q2]}  ➜  incoming list  {q:[p1,p2]}\n",
    "H_out = {z: [] for z in range(len(clusters))}\n",
    "for src, tgts in similarity_graph.items():\n",
    "    for t in tgts:\n",
    "        H_out[t].append(src)        # cluster src  ➜  learns from  ➜  t"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{0: [0]}"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# clusters = hierarchical_clustering(copy.deepcopy(adj_mat), thresh=args.cluster_alpha, linkage='average')\n",
    "similarity_graph"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO:matplotlib.mathtext:Substituting symbol L from STIXNonUnicode\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n",
      "INFO:matplotlib.mathtext:Substituting symbol L from STIXNonUnicode\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n",
      "Breaking HC\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO:matplotlib.mathtext:Substituting symbol L from STIXNonUnicode\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n",
      "INFO:matplotlib.mathtext:Substituting symbol L from STIXNonUnicode\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n",
      "INFO:matplotlib.mathtext:Substituting symbol L from STIXNonUnicode\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n"
     ]
    },
    {
     "data": {
      "image/png": "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Dg5WANmTsHnz5qYkIiIiQlm+fHmh16jb7lSvXt1mm0ePHlWqVatm+oXZtGlTJTEx0fSf8tFHH/WZWHv27KmUL1/edFP3XixVqpTV45a/ALwdZ926dU1fu9atWyvt2rWze3OG1nFWr17d9D2vX7++0rp1a6VJkyZKVFSUKf6WLVsqZ86ccSpOT8RaFFcTO0/EqcYSHBys1KlTR7n55puVBg0amK5hMBiU0aNH6x6nokgCf/fdd5u+13Xq1FFatGhh2sYqMjLS4T1NPR2rqn379goA5b777nM6Lk/HmZubq/Tr18/09QwPD1caNmyo1K1b15SAAM7vzeuJr6f6s5iQkKC0bNlSqVmzpim+Bg0aKCkpKU7FqCiKcuzYMavfwWXKlDG1afl4z549nYrVU59P5B4mdhr766+/lN69eysVK1ZUwsLClISEBGXw4MHKvn37bJ7vyC/NS5cuKaNHj1bq1q2rhIeHK2XLllU6duyozJkzx6diVT84i7s5u0+nlnGqCZMjN2dpGeeSJUuUp59+WmnRooUSHx+vhIaGKhEREUpiYqJy3333KXPmzHFreMMTP6e2uJPYaR3n4sWLlccff1y56aablEqVKilhYWFKZGSkUrduXeWRRx5RNm/e7FKMWsepys/PV2bOnKnccsstStmyZZWwsDAlMTFReeyxx5QDBw74VKyWe5j+/PPPLsfm6TgXLVqk9OrVS0lISFDCwsIUo9GoJCYmKv3791fWrl3rE3EmJSUpHTt2VCpVqqSEhoYq5cuXVzp06KB88MEHDk8RKOjIkSMO/c7r2LGjU7Eqiuc+n8h1BkWxs1abiIiIiPwKV8USERERBQgmdkREREQBgokdERERUYBgYkdEREQUIJjYEREREQUIJnZEREREAYKJHREREVGAYGJHREREFCCY2BEREREFCCZ2RERERAGCiR35rVmzZsFgMBS6RUREoFKlSmjYsCEefPBBvP322zh+/Hix7a1evdrURkpKiuffAHlESfg++sJ7TElJMcWwevVql9rQ8n0sXrwYBoMBbdq0casde86fP49SpUqhVKlSOH36tEeuQaQFJnYUcK5du4azZ89i9+7dmDNnDp5//nnUqFEDvXv3xqlTp7waixYffiSSk5NhMBiQmJiodyjkY3JzczF69GgA8nPiCbGxsRg+fDgyMjKQlJTkkWsQaYGJHQWEn3/+Genp6UhPT8elS5eQkpKCdevWYcqUKahXrx7y8vKwcOFCNG7cGBs3btQ7XCLS0Oeff469e/eidevWuOOOOzx2nRdeeAFRUVH44osvsHv3bo9dh8gdTOwoIERERJiGScqUKYPq1aujffv2GDVqFPbs2YO3334bwcHBSEtLwz333IMTJ04UaqNTp05QFAWKorBXiMhP5OfnY/LkyQCAZ5991qPXiouLwwMPPIC8vDy8/vrrHr0WkauY2FHAMxgMGDFiBKZMmQIAOHfunMeGa4jIu3755RccO3YMpUqVQq9evTx+vQEDBgAAFixYgLS0NI9fj8hZTOyoxHj++edRt25dAMBXX32F1NRUq+eLm8idk5ODjz/+GLfeeivi4uIQGhqKcuXKoV69eujRowemT5+O8+fPm85PTExEjRo1TP++9dZbCy30sLzOtWvX8Msvv2DYsGG48cYbUbp0aYSGhqJChQro2rUrPvvsM1y/ft3u+xsyZAgMBgM6deoEANi+fTsGDhyIKlWqwGg0okqVKhgyZAgOHTpU7Nfq8uXLeOONN9ChQwdUqFDB9Pr27dvj9ddfx+HDh+2+dsuWLXjkkUdQu3ZtREVFITo6GjfeeCPGjh1r9fVxlPp9mTBhAgDg6NGjhb6O6nu2JSMjA6+++ioaNWqEqKgolC1bFp06dcL8+fPtvqbgfL6dO3fikUceQY0aNWA0GlG2bFnN3rezP1davUdVeno6Jk+ejJtvvhkxMTEIDw9HtWrVMGDAAPzxxx/Fvr44S5YswW233YZy5cohKioKjRo1wquvvorMzEy32waATz/9FABw3333ISIioshzT506hWeeeQa1atUyvc977rkHmzZtMp0zadIkGAwGPPTQQzbb6NixI6pUqYJr167hyy+/1OQ9EGlKIfJTM2fOVAAoAJTff//doddMmTLF9Jr58+dbPff777+bnjty5IjVc+np6Urr1q1Nz9u7zZs3z/Sa6tWrF3u+5XWee+65Ys9v3bq1cvHiRZvvbfDgwQoApWPHjsqcOXMUo9Fos42YmBhl+/btdr9GK1euVGJjY4uMo1evXoVel5eXp4wcOVIxGAx2XxcbG6v8+eefxX6fLFl+X+zdOnbsaPP8jRs3KjfccIPd102cONHmNZOSkhQASvXq1ZVFixYp4eHhVq8rU6aMJu/blZ8rrd6joijK9u3blYSEhCKv/cILLyj5+fmFXnvkyJFi//+NHDnSbruNGzdWFi5caPf/nCOuXr2qhIaGKgCU7777rshzf/vtN6V06dI2YwkKClJ+/vlnRVEU5cYbb1QAKAsXLrTb1tChQxUAStu2bZ2OmcjTmNiR33IlsVu7dq3pNc8//7zVc0Uldi+//LICQAkODlbGjRunbN26VTl79qxy/PhxZePGjcrHH3+sdOjQQVmwYIHpNRkZGcquXbtMbf78889Kenq61c3yA/Pll19W+vbtq3z99dfK5s2blePHjytnz55V/vnnHyU5OVkpV66cAkB54IEHbL43NbFLSEhQjEaj0qFDB2X58uXKuXPnlOPHjyvvvvuuEhYWpgBQ2rRpY7ONv/76y3ROTEyMMnnyZGXnzp1KWlqacuzYMWXJkiXKo48+qvTv37/Qa1944QXTe33ooYeUtWvXKqmpqcqZM2eURYsWKY0aNVIAKHFxccqpU6cc+n4piqLk5uYq6enpypgxYxQASrVq1Qp9HTMzM21+H2vWrKnExsYqH3zwgXL48GHl/PnzyooVK0yxBAcHK7t37y50TTWxK126tBIdHa00bNhQWbBggXL69GnlxIkTyg8//KDJ+3bl50qr93j+/HklPj5eAaBEREQokydPVg4ePKikpqYqK1asUNq1a2e6xtSpUwu9vrjE7rPPPjM937JlS+W3335TUlNTlQMHDijJyclKaGioUqNGDbcSu99++830+oMHD9o9b+/evUpkZKQCQKlbt67y22+/KefOnVPWr1+vNGvWzPT92blzpwJAiYyMVDIyMuy29/HHHysAlLCwMKufPSJfwMSO/JYrid2ZM2dMrymYnBSV2N10000KAOW5555zKkZHejUctWPHDiUkJEQxGAw2P8TUxA6Acueddyo5OTmFzpk2bZrpnD179lg9l5+fb+r5iY2NVfbv3283loJtb9myxdRj9frrr9t8TXp6uqn94cOHO/KWrVj2ohXF8vtYqlSpQu9TURTlxIkTSkREhAJAGTVqlN1rqYnApUuXbF7L3fft6s+VFu/x2WefVQAoBoNBWbZsWaHns7Ozlfbt2ysAFKPRqJw9e9bq+aJ+trOyspTy5csrAJQmTZrYTJJmz55t1WvmSmL3yiuvKACUcuXKFXlenz59TEluwZ/r48ePK1FRUQoA0/u99957i2xv69atprhX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",
      "text/plain": [
       "<Figure size 640x480 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import os\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import copy\n",
    "from scipy.cluster.hierarchy import linkage, fcluster\n",
    "from scipy.spatial.distance import squareform\n",
    "\n",
    "\n",
    "# 2. Loss function (unchanged)\n",
    "def clustering_loss_with_tiny_penalty(clusters, adjacency_matrix, gamma=1.0, tau=1.0):\n",
    "    intra_sum = 0.0\n",
    "    sizes = [len(c) for c in clusters if len(c) > 0]\n",
    "    K = len(sizes)\n",
    "    n = sum(sizes)\n",
    "\n",
    "    # Intra-cluster compactness term\n",
    "    for cluster in clusters:\n",
    "        size = len(cluster)\n",
    "        if size == 0:\n",
    "            continue\n",
    "        dist = sum(adjacency_matrix[i, j] for i in cluster for j in cluster)\n",
    "        intra_sum += dist / (size * size)\n",
    "\n",
    "    # Unbalanced cluster penalty\n",
    "    s_bar = n / K\n",
    "    sigma_s = np.std(sizes, ddof=0)\n",
    "    s_thresh = s_bar - gamma * sigma_s\n",
    "\n",
    "    penalty = np.mean([\n",
    "        np.exp((max(0, s_thresh - s_c)) / tau)\n",
    "        for s_c in sizes\n",
    "    ])\n",
    "\n",
    "  \n",
    "    return intra_sum + 0.01*penalty, intra_sum, penalty\n",
    "\n",
    "\n",
    "# Grid search over alpha\n",
    "alpha_grid = np.linspace(0.05, 1.00, 20)\n",
    "gamma_penalty = 1.0\n",
    "tau_penalty = 1.0\n",
    "cluster_counts = []\n",
    "cluster_losses = []\n",
    "\n",
    "for a in alpha_grid:\n",
    "    clusters = hierarchical_clustering(copy.deepcopy(adj_mat), thresh=a, linkage='average')\n",
    "    score, intra, penalty = clustering_loss_with_tiny_penalty(clusters, copy.deepcopy(adj_mat), gamma_penalty, tau_penalty)\n",
    "    cluster_counts.append(len(clusters))\n",
    "    cluster_losses.append(min(score,1))\n",
    "\n",
    "# Plot the figure with larger font sizes\n",
    "fig, ax1 = plt.subplots()\n",
    "\n",
    "# Right y-axis (number of clusters)\n",
    "ax2 = ax1.twinx()\n",
    "ax2.bar(alpha_grid, cluster_counts, width=0.03, color='blue', alpha=0.7)\n",
    "ax2.set_ylabel('Number of clusters', color='blue', fontsize=19)\n",
    "ax2.tick_params(axis='y', labelcolor='blue', labelsize=17)\n",
    "\n",
    "# Left y-axis (clustering loss)\n",
    "ax1.plot(alpha_grid, cluster_losses, color='red', marker='x', linestyle='--')\n",
    "ax1.set_ylabel(r'Clustering loss $\\mathcal{L}_{\\mathbb{C}}$', color='red', fontsize=19)\n",
    "ax1.tick_params(axis='y', labelcolor='red', labelsize=17)\n",
    "\n",
    "# X-axis (alpha)\n",
    "ax1.set_xlabel('Distance threshold (α)', fontsize=19)\n",
    "ax1.set_xticks(np.arange(0.0, 1.05, 0.1))\n",
    "ax1.tick_params(axis='x', labelsize=17)\n",
    "\n",
    "plt.grid(True)\n",
    "plt.tight_layout()\n",
    "\n",
    "# Save plot and data\n",
    "os.makedirs(\"plots\", exist_ok=True)\n",
    "plt.savefig(\"plots/\"+args.dataset+\"_.png\", dpi=300, bbox_inches='tight')\n",
    "np.savez(\n",
    "    f\"plots/{args.dataset}_plot_data.npz\",\n",
    "    alpha_grid=alpha_grid,\n",
    "    cluster_counts=np.array(cluster_counts),\n",
    "    cluster_losses=np.array(cluster_losses)\n",
    ")\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Cluster 0, First Client 0: {4: 21, 7: 1, 8: 214}\n",
      "Cluster 1, First Client 6: {2: 153, 3: 79, 9: 123}\n",
      "Cluster 2, First Client 7: {5: 368, 6: 118, 9: 48}\n",
      "Cluster 3, First Client 19: {1: 475, 2: 72, 8: 28}\n"
     ]
    }
   ],
   "source": [
    "for k in range(len(clusters)):\n",
    "    print(f\"Cluster {k}, First Client {clusters[k][0]}:\", traindata_cls_counts[clusters[k][0]])\n",
    "     "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      ">>> All clusters switched to CombinedModelMNIST\n"
     ]
    }
   ],
   "source": [
    "#Goes to utility\n",
    "def clone_encoder(encoder_template: nn.Module) -> nn.Module:\n",
    "    \"\"\"\n",
    "    Return a fresh deep copy of the encoder (with the same structure + random weights).\n",
    "    \"\"\"\n",
    "    return copy.deepcopy(encoder_template).to(next(encoder_template.parameters()).device)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "# ------------------------------------------------------------\n",
    "# Robust FedAvg over a list of encoders (any architecture)\n",
    "# ------------------------------------------------------------\n",
    "def fedavg_delta(delta_list, weights):\n",
    "    \"\"\"\n",
    "    FedAvg for *additive updates*  Δθ.\n",
    "    Keys are already identical across dicts.\n",
    "    \"\"\"\n",
    "    tot = float(sum(weights))\n",
    "    out = copy.deepcopy(delta_list[0])\n",
    "    with torch.no_grad():\n",
    "        for k in out:\n",
    "            out[k].zero_()\n",
    "            for w, d in zip(weights, delta_list):\n",
    "                out[k] += (w / tot) * d[k]\n",
    "    return out\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "def fedavg_encoder(encoder_list: list[nn.Module]) -> nn.Module:\n",
    "    \"\"\"\n",
    "    Return a copy of encoder_list[0] whose parameters are the\n",
    "    element-wise average of all encoders in the list.\n",
    "    \"\"\"\n",
    "    n = len(encoder_list)\n",
    "    assert n > 0, \"encoder_list must be non-empty\"\n",
    "\n",
    "    # clone first encoder as template\n",
    "    avg_enc = copy.deepcopy(encoder_list[0])\n",
    "    avg_sd  = avg_enc.state_dict()           # OrderedDict of tensors\n",
    "\n",
    "    with torch.no_grad():\n",
    "        for k in avg_sd.keys():\n",
    "            stacked = torch.stack([enc.state_dict()[k] for enc in encoder_list], dim=0)\n",
    "            avg_sd[k].copy_(stacked.mean(dim=0))\n",
    "\n",
    "    avg_enc.load_state_dict(avg_sd)\n",
    "    return avg_enc\n",
    "\n",
    "\n",
    "def make_combined_model_from_single(single_model: SimpleCNNMNIST2,\n",
    "                                    cluster_enc: nn.Module,\n",
    "                                    num_classes: int = 10) -> CombinedModelMNIST:\n",
    "    \"\"\"\n",
    "    single_model : an *instance* of SimpleCNNMNIST2 (for architecture reference)\n",
    "    cluster_enc  : averaged encoder to use as PRIMARY encoder\n",
    "    \"\"\"\n",
    "    # fresh secondary encoder (same dim 84-D) – random init\n",
    "    sec_enc = clone_encoder(single_model.encoder)\n",
    "    # empty classifier state-dict with correct input dim 168\n",
    "    new_clf = nn.Linear(168, num_classes)\n",
    "    clf_sd  = new_clf.state_dict()          # random weights\n",
    "    return CombinedModelMNIST(cluster_enc, sec_enc, clf_sd, num_classes)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "def switch_cluster_to_combined(\n",
    "    clients, \n",
    "    cluster_ids, \n",
    "    w_glob_per_cluster, \n",
    "    cluster_id, \n",
    "    num_classes=10\n",
    "):\n",
    "    \"\"\"\n",
    "    Initialize combined model for a cluster using FedAvg on primary encoders.\n",
    "    \n",
    "    clients             : list[Client_ClusterFL]\n",
    "    cluster_ids         : list[int] – client indices in the SAME cluster\n",
    "    w_glob_per_cluster  : list of cluster-level global state_dicts\n",
    "    cluster_id          : int – ID of the current cluster\n",
    "    num_classes         : int – number of output classes\n",
    "    \"\"\"\n",
    "    # 1. collect trained encoders from clients\n",
    "    encoders = [copy.deepcopy(clients[i].net.encoder).to(clients[i].device)\n",
    "                for i in cluster_ids]\n",
    "\n",
    "    # 2. FedAvg on primary encoder\n",
    "    cluster_enc = fedavg_encoder(encoders)\n",
    "\n",
    "    # 3. use the first client’s model as a template\n",
    "    single_template = clients[cluster_ids[0]].net\n",
    "\n",
    "    # 4. build the combined model (primary + frozen secondary + classifier)\n",
    "    combined = make_combined_model_from_single(\n",
    "        single_model=single_template,\n",
    "        cluster_enc=cluster_enc,\n",
    "        num_classes=num_classes\n",
    "    )\n",
    "\n",
    "    # 5. get state_dict of combined model\n",
    "    combined_sd = combined.state_dict()\n",
    "\n",
    "    # 6. assign combined model to all clients in the cluster\n",
    "    for i in cluster_ids:\n",
    "        clients[i].net = copy.deepcopy(combined)\n",
    "        clients[i].net.load_state_dict(combined_sd)\n",
    "\n",
    "    # 7. store cluster-level global model\n",
    "    w_glob_per_cluster[cluster_id] = copy.deepcopy(combined_sd)\n",
    "\n",
    "\n",
    "\n",
    "w_glob_per_cluster = [None] * len(clusters)\n",
    "\n",
    "for z, clist in enumerate(clusters):\n",
    "    switch_cluster_to_combined(\n",
    "        clients=clients,\n",
    "        cluster_ids=clist,\n",
    "        w_glob_per_cluster=w_glob_per_cluster,\n",
    "        cluster_id=z,\n",
    "        num_classes=10\n",
    "    )\n",
    "print(\">>> All clusters switched to CombinedModelMNIST\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "###################################### Clustered FL training \n",
    "# ============================================================\n",
    "# Helpers\n",
    "# ============================================================\n",
    "\n",
    "from collections import OrderedDict\n",
    "\n",
    "def cluster_participants(idxs_users, clients_clust_id):\n",
    "    \"\"\"\n",
    "    Map sampled user IDs to {cluster_id: [user_ids]}.\n",
    "    \"\"\"\n",
    "    out = {}\n",
    "    for uid in idxs_users:\n",
    "        cid = clients_clust_id[uid]\n",
    "        out.setdefault(cid, []).append(uid)\n",
    "    return out\n",
    "\n",
    "\n",
    "def make_weight_vec(id_list, net_dataidx_map):\n",
    "    \"\"\"\n",
    "    Return FedAvg weights proportional to each client's data size.\n",
    "    \"\"\"\n",
    "    sizes = [len(net_dataidx_map[u]) for u in id_list]\n",
    "    tot   = sum(sizes)\n",
    "    return [s / tot for s in sizes]\n",
    "\n",
    "\n",
    "def FedAvg_trainable(sd_list, weights):\n",
    "    \"\"\"\n",
    "    FedAvg only the trainable blocks: own_encoder.* and classifier.*.\n",
    "    Frozen secondary_* keys are left untouched.\n",
    "    \"\"\"\n",
    "    out = copy.deepcopy(sd_list[0])\n",
    "    with torch.no_grad():\n",
    "        for k in out:\n",
    "            if k.startswith((\"own_encoder\", \"classifier\")):\n",
    "                out[k].zero_()\n",
    "                for w, sd in zip(weights, sd_list):\n",
    "                    out[k] += w * sd[k]\n",
    "    return out\n",
    "\n",
    "def fedavg_secondary(sd_list, weights):\n",
    "    tot = float(sum(weights))\n",
    "    out = copy.deepcopy(sd_list[0])\n",
    "    with torch.no_grad():\n",
    "        for k in out:                       # keys already 'secondary_encoder.*'\n",
    "            out[k].zero_()\n",
    "            for w, sd in zip(weights, sd_list):\n",
    "                out[k] += (w / tot) * sd[k]\n",
    "    return out\n",
    "\n",
    "# ============================================================\n",
    "# Metric bookkeeping\n",
    "# ============================================================\n",
    "\n",
    "client_metric = {\n",
    "    uid: {\"best_pre\": 0.0, \"best_post\": 0.0, \"best_any\": 0.0}\n",
    "    for uid in range(args.num_users)\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "##### ROUND 1 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.4970\n",
      "avg best PRE   : 2.99%\n",
      "avg best POST  : 16.80%\n",
      "avg best ANY   : 16.80%\n",
      "\n",
      "Client   0 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   1 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   2 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   3 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   4 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   5 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   6 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   7 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   8 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   9 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  10 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  11 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  12 | best_pre=  0.00 | best_post= 96.67 | best_any= 96.67\n",
      "Client  13 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  14 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  15 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  16 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  17 | best_pre= 33.13 | best_post= 95.27 | best_any= 95.27\n",
      "Client  18 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  19 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  20 | best_pre=  0.00 | best_post= 96.77 | best_any= 96.77\n",
      "Client  21 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  22 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  23 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  24 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  25 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  26 | best_pre= 33.13 | best_post= 91.60 | best_any= 91.60\n",
      "Client  27 | best_pre=  0.00 | best_post= 92.57 | best_any= 92.57\n",
      "Client  28 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  29 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  30 | best_pre= 33.13 | best_post= 91.33 | best_any= 91.33\n",
      "Client  31 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  32 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  33 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  34 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  35 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  36 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  37 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  38 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  39 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  40 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  41 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  42 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  43 | best_pre= 33.27 | best_post= 97.07 | best_any= 97.07\n",
      "Client  44 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  45 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  46 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  47 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  48 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  49 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  50 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  51 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  52 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  53 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  54 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  55 | best_pre=  0.00 | best_post= 93.20 | best_any= 93.20\n",
      "Client  56 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  57 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  58 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  59 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  60 | best_pre=  0.00 | best_post= 79.93 | best_any= 79.93\n",
      "Client  61 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  62 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  63 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  64 | best_pre= 33.27 | best_post= 80.93 | best_any= 80.93\n",
      "Client  65 | best_pre= 33.13 | best_post= 93.43 | best_any= 93.43\n",
      "Client  66 | best_pre=  0.00 | best_post= 67.30 | best_any= 67.30\n",
      "Client  67 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  68 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  69 | best_pre=  0.00 | best_post= 76.27 | best_any= 76.27\n",
      "Client  70 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  71 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  72 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  73 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  74 | best_pre= 33.27 | best_post= 61.40 | best_any= 61.40\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  76 | best_pre= 33.27 | best_post= 55.70 | best_any= 55.70\n",
      "Client  77 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  78 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  79 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  80 | best_pre= 33.13 | best_post= 44.20 | best_any= 44.20\n",
      "Client  81 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  82 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  83 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  84 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  85 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  86 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  87 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  88 | best_pre=  0.00 | best_post= 92.60 | best_any= 92.60\n",
      "Client  89 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  90 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  91 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  92 | best_pre=  0.00 | best_post= 93.60 | best_any= 93.60\n",
      "Client  93 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  94 | best_pre=  0.00 | best_post= 87.53 | best_any= 87.53\n",
      "Client  95 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  96 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  97 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  98 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  99 | best_pre=  0.00 | best_post= 92.33 | best_any= 92.33\n",
      "\n",
      "##### ROUND 2 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.1514\n",
      "avg best PRE   : 12.41%\n",
      "avg best POST  : 31.79%\n",
      "avg best ANY   : 31.82%\n",
      "\n",
      "Client   0 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   1 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   2 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   3 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   4 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   5 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   6 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   7 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   8 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   9 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  10 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  11 | best_pre=  9.53 | best_post= 84.13 | best_any= 84.13\n",
      "Client  12 | best_pre= 47.10 | best_post= 98.07 | best_any= 98.07\n",
      "Client  13 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  14 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  15 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  16 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  17 | best_pre= 33.13 | best_post= 95.27 | best_any= 95.27\n",
      "Client  18 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  19 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  20 | best_pre=  0.00 | best_post= 96.77 | best_any= 96.77\n",
      "Client  21 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  22 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  23 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  24 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  25 | best_pre= 47.10 | best_post= 97.30 | best_any= 97.30\n",
      "Client  26 | best_pre= 94.83 | best_post= 94.50 | best_any= 94.83\n",
      "Client  27 | best_pre=  0.00 | best_post= 92.57 | best_any= 92.57\n",
      "Client  28 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  29 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  30 | best_pre= 33.13 | best_post= 91.33 | best_any= 91.33\n",
      "Client  31 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  32 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  33 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  34 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  35 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  36 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  37 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  38 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  39 | best_pre=  0.00 | best_post= 97.93 | best_any= 97.93\n",
      "Client  40 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  41 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  42 | best_pre= 94.83 | best_post= 94.53 | best_any= 94.83\n",
      "Client  43 | best_pre= 33.27 | best_post= 97.07 | best_any= 97.07\n",
      "Client  44 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  45 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  46 | best_pre= 94.83 | best_post= 95.80 | best_any= 95.80\n",
      "Client  47 | best_pre=  9.53 | best_post= 92.90 | best_any= 92.90\n",
      "Client  48 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  49 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  50 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  51 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  52 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  53 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  54 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  55 | best_pre=  0.00 | best_post= 93.20 | best_any= 93.20\n",
      "Client  56 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  57 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  58 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  59 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  60 | best_pre=  0.00 | best_post= 79.93 | best_any= 79.93\n",
      "Client  61 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  62 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  63 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  64 | best_pre= 33.27 | best_post= 80.93 | best_any= 80.93\n",
      "Client  65 | best_pre= 33.13 | best_post= 93.43 | best_any= 93.43\n",
      "Client  66 | best_pre= 42.87 | best_post= 79.43 | best_any= 79.43\n",
      "Client  67 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  68 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  69 | best_pre=  0.00 | best_post= 76.27 | best_any= 76.27\n",
      "Client  70 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  71 | best_pre= 47.10 | best_post= 97.73 | best_any= 97.73\n",
      "Client  72 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  73 | best_pre=  9.53 | best_post= 59.10 | best_any= 59.10\n",
      "Client  74 | best_pre= 33.27 | best_post= 61.40 | best_any= 61.40\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  76 | best_pre= 33.27 | best_post= 55.70 | best_any= 55.70\n",
      "Client  77 | best_pre= 94.83 | best_post= 94.83 | best_any= 94.83\n",
      "Client  78 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  79 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  80 | best_pre= 33.13 | best_post= 44.20 | best_any= 44.20\n",
      "Client  81 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  82 | best_pre= 94.83 | best_post= 95.23 | best_any= 95.23\n",
      "Client  83 | best_pre=  0.00 | best_post= 93.20 | best_any= 93.20\n",
      "Client  84 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  85 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  86 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  87 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  88 | best_pre=  0.00 | best_post= 92.60 | best_any= 92.60\n",
      "Client  89 | best_pre= 47.10 | best_post= 96.43 | best_any= 96.43\n",
      "Client  90 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  91 | best_pre= 47.10 | best_post= 97.53 | best_any= 97.53\n",
      "Client  92 | best_pre=  0.00 | best_post= 93.60 | best_any= 93.60\n",
      "Client  93 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  94 | best_pre=  0.00 | best_post= 87.53 | best_any= 87.53\n",
      "Client  95 | best_pre= 94.83 | best_post= 92.53 | best_any= 94.83\n",
      "Client  96 | best_pre=  9.53 | best_post= 92.60 | best_any= 92.60\n",
      "Client  97 | best_pre= 47.10 | best_post= 97.20 | best_any= 97.20\n",
      "Client  98 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  99 | best_pre= 42.87 | best_post= 95.83 | best_any= 95.83\n",
      "\n",
      "##### ROUND 3 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.0778\n",
      "avg best PRE   : 24.01%\n",
      "avg best POST  : 41.99%\n",
      "avg best ANY   : 42.09%\n",
      "\n",
      "Client   0 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   1 | best_pre=  7.80 | best_post= 91.87 | best_any= 91.87\n",
      "Client   2 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   3 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   4 | best_pre= 39.30 | best_post= 92.30 | best_any= 92.30\n",
      "Client   5 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   6 | best_pre= 97.10 | best_post= 96.07 | best_any= 97.10\n",
      "Client   7 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   8 | best_pre= 39.30 | best_post= 66.40 | best_any= 66.40\n",
      "Client   9 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  10 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  11 | best_pre=  9.53 | best_post= 84.13 | best_any= 84.13\n",
      "Client  12 | best_pre= 97.10 | best_post= 98.40 | best_any= 98.40\n",
      "Client  13 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  14 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  15 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  16 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  17 | best_pre= 33.13 | best_post= 95.27 | best_any= 95.27\n",
      "Client  18 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  19 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  20 | best_pre= 97.10 | best_post= 97.73 | best_any= 97.73\n",
      "Client  21 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  22 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  23 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  24 | best_pre=  7.80 | best_post= 90.27 | best_any= 90.27\n",
      "Client  25 | best_pre= 97.10 | best_post= 98.17 | best_any= 98.17\n",
      "Client  26 | best_pre= 95.80 | best_post= 94.90 | best_any= 95.80\n",
      "Client  27 | best_pre= 39.30 | best_post= 96.70 | best_any= 96.70\n",
      "Client  28 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  29 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  30 | best_pre= 33.13 | best_post= 91.33 | best_any= 91.33\n",
      "Client  31 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  32 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  33 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  34 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  35 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  36 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  37 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  38 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  39 | best_pre=  0.00 | best_post= 97.93 | best_any= 97.93\n",
      "Client  40 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  41 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  42 | best_pre= 94.83 | best_post= 94.53 | best_any= 94.83\n",
      "Client  43 | best_pre= 33.27 | best_post= 97.07 | best_any= 97.07\n",
      "Client  44 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  45 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  46 | best_pre= 94.83 | best_post= 95.80 | best_any= 95.80\n",
      "Client  47 | best_pre=  9.53 | best_post= 92.90 | best_any= 92.90\n",
      "Client  48 | best_pre= 97.10 | best_post= 98.50 | best_any= 98.50\n",
      "Client  49 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  50 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  51 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  52 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  53 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  54 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  55 | best_pre= 97.10 | best_post= 97.93 | best_any= 97.93\n",
      "Client  56 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  57 | best_pre= 97.10 | best_post= 94.70 | best_any= 97.10\n",
      "Client  58 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  59 | best_pre= 95.80 | best_post= 95.93 | best_any= 95.93\n",
      "Client  60 | best_pre=  0.00 | best_post= 79.93 | best_any= 79.93\n",
      "Client  61 | best_pre= 39.30 | best_post= 90.77 | best_any= 90.77\n",
      "Client  62 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  63 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  64 | best_pre= 33.27 | best_post= 80.93 | best_any= 80.93\n",
      "Client  65 | best_pre= 95.80 | best_post= 95.67 | best_any= 95.80\n",
      "Client  66 | best_pre= 42.87 | best_post= 79.43 | best_any= 79.43\n",
      "Client  67 | best_pre= 95.80 | best_post= 95.07 | best_any= 95.80\n",
      "Client  68 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  69 | best_pre=  0.00 | best_post= 76.27 | best_any= 76.27\n",
      "Client  70 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  71 | best_pre= 97.10 | best_post= 97.73 | best_any= 97.73\n",
      "Client  72 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  73 | best_pre=  9.53 | best_post= 59.10 | best_any= 59.10\n",
      "Client  74 | best_pre= 33.27 | best_post= 61.40 | best_any= 61.40\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  76 | best_pre= 33.27 | best_post= 55.70 | best_any= 55.70\n",
      "Client  77 | best_pre= 95.80 | best_post= 94.83 | best_any= 95.80\n",
      "Client  78 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  79 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  80 | best_pre= 33.13 | best_post= 44.20 | best_any= 44.20\n",
      "Client  81 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  82 | best_pre= 94.83 | best_post= 95.23 | best_any= 95.23\n",
      "Client  83 | best_pre=  0.00 | best_post= 93.20 | best_any= 93.20\n",
      "Client  84 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  85 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  86 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  87 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  88 | best_pre=  0.00 | best_post= 92.60 | best_any= 92.60\n",
      "Client  89 | best_pre= 47.10 | best_post= 96.43 | best_any= 96.43\n",
      "Client  90 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  91 | best_pre= 47.10 | best_post= 97.53 | best_any= 97.53\n",
      "Client  92 | best_pre=  0.00 | best_post= 93.60 | best_any= 93.60\n",
      "Client  93 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  94 | best_pre=  0.00 | best_post= 87.53 | best_any= 87.53\n",
      "Client  95 | best_pre= 94.83 | best_post= 92.53 | best_any= 94.83\n",
      "Client  96 | best_pre=  9.53 | best_post= 92.60 | best_any= 92.60\n",
      "Client  97 | best_pre= 47.10 | best_post= 97.20 | best_any= 97.20\n",
      "Client  98 | best_pre= 95.80 | best_post= 94.40 | best_any= 95.80\n",
      "Client  99 | best_pre= 42.87 | best_post= 95.83 | best_any= 95.83\n",
      "\n",
      "##### ROUND 4 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.0954\n",
      "avg best PRE   : 32.46%\n",
      "avg best POST  : 53.82%\n",
      "avg best ANY   : 53.97%\n",
      "\n",
      "Client   0 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   1 | best_pre= 10.13 | best_post= 91.87 | best_any= 91.87\n",
      "Client   2 | best_pre=  0.00 | best_post= 98.53 | best_any= 98.53\n",
      "Client   3 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   4 | best_pre= 39.30 | best_post= 92.30 | best_any= 92.30\n",
      "Client   5 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   6 | best_pre= 97.10 | best_post= 96.07 | best_any= 97.10\n",
      "Client   7 | best_pre= 96.03 | best_post= 95.63 | best_any= 96.03\n",
      "Client   8 | best_pre= 39.30 | best_post= 66.40 | best_any= 66.40\n",
      "Client   9 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  10 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  11 | best_pre=  9.53 | best_post= 84.13 | best_any= 84.13\n",
      "Client  12 | best_pre= 97.10 | best_post= 98.40 | best_any= 98.40\n",
      "Client  13 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  14 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  15 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  16 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  17 | best_pre= 33.13 | best_post= 95.27 | best_any= 95.27\n",
      "Client  18 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  19 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  20 | best_pre= 97.10 | best_post= 97.73 | best_any= 97.73\n",
      "Client  21 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  22 | best_pre= 10.13 | best_post= 93.93 | best_any= 93.93\n",
      "Client  23 | best_pre=  0.00 | best_post= 99.00 | best_any= 99.00\n",
      "Client  24 | best_pre=  7.80 | best_post= 90.27 | best_any= 90.27\n",
      "Client  25 | best_pre= 97.10 | best_post= 98.17 | best_any= 98.17\n",
      "Client  26 | best_pre= 95.80 | best_post= 94.90 | best_any= 95.80\n",
      "Client  27 | best_pre= 39.30 | best_post= 96.70 | best_any= 96.70\n",
      "Client  28 | best_pre= 96.03 | best_post= 95.00 | best_any= 96.03\n",
      "Client  29 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  30 | best_pre= 33.13 | best_post= 91.33 | best_any= 91.33\n",
      "Client  31 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  32 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  33 | best_pre= 10.13 | best_post= 84.43 | best_any= 84.43\n",
      "Client  34 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  35 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  36 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  37 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  38 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  39 | best_pre=  0.00 | best_post= 97.93 | best_any= 97.93\n",
      "Client  40 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  41 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  42 | best_pre= 96.03 | best_post= 94.67 | best_any= 96.03\n",
      "Client  43 | best_pre= 33.27 | best_post= 97.07 | best_any= 97.07\n",
      "Client  44 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  45 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  46 | best_pre= 94.83 | best_post= 95.80 | best_any= 95.80\n",
      "Client  47 | best_pre=  9.53 | best_post= 92.90 | best_any= 92.90\n",
      "Client  48 | best_pre= 97.10 | best_post= 98.50 | best_any= 98.50\n",
      "Client  49 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  50 | best_pre= 42.63 | best_post= 86.67 | best_any= 86.67\n",
      "Client  51 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  52 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  53 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  54 | best_pre= 42.63 | best_post= 95.00 | best_any= 95.00\n",
      "Client  55 | best_pre= 97.10 | best_post= 97.93 | best_any= 97.93\n",
      "Client  56 | best_pre= 98.03 | best_post= 98.33 | best_any= 98.33\n",
      "Client  57 | best_pre= 97.10 | best_post= 94.70 | best_any= 97.10\n",
      "Client  58 | best_pre= 98.03 | best_post= 97.00 | best_any= 98.03\n",
      "Client  59 | best_pre= 95.80 | best_post= 95.93 | best_any= 95.93\n",
      "Client  60 | best_pre= 42.63 | best_post= 96.87 | best_any= 96.87\n",
      "Client  61 | best_pre= 39.30 | best_post= 90.77 | best_any= 90.77\n",
      "Client  62 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  63 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  64 | best_pre= 33.27 | best_post= 97.00 | best_any= 97.00\n",
      "Client  65 | best_pre= 95.80 | best_post= 95.67 | best_any= 95.80\n",
      "Client  66 | best_pre= 42.87 | best_post= 79.43 | best_any= 79.43\n",
      "Client  67 | best_pre= 95.80 | best_post= 95.07 | best_any= 95.80\n",
      "Client  68 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  69 | best_pre=  0.00 | best_post= 76.27 | best_any= 76.27\n",
      "Client  70 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  71 | best_pre= 97.10 | best_post= 97.73 | best_any= 97.73\n",
      "Client  72 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  73 | best_pre=  9.53 | best_post= 59.10 | best_any= 59.10\n",
      "Client  74 | best_pre= 33.27 | best_post= 61.40 | best_any= 61.40\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  76 | best_pre= 33.27 | best_post= 55.70 | best_any= 55.70\n",
      "Client  77 | best_pre= 95.80 | best_post= 94.83 | best_any= 95.80\n",
      "Client  78 | best_pre= 98.03 | best_post= 95.40 | best_any= 98.03\n",
      "Client  79 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  80 | best_pre= 33.13 | best_post= 44.20 | best_any= 44.20\n",
      "Client  81 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  82 | best_pre= 94.83 | best_post= 95.23 | best_any= 95.23\n",
      "Client  83 | best_pre=  0.00 | best_post= 97.97 | best_any= 97.97\n",
      "Client  84 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  85 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  86 | best_pre= 10.13 | best_post= 88.80 | best_any= 88.80\n",
      "Client  87 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  88 | best_pre= 98.03 | best_post= 98.23 | best_any= 98.23\n",
      "Client  89 | best_pre= 47.10 | best_post= 96.43 | best_any= 96.43\n",
      "Client  90 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  91 | best_pre= 47.10 | best_post= 97.53 | best_any= 97.53\n",
      "Client  92 | best_pre=  0.00 | best_post= 93.60 | best_any= 93.60\n",
      "Client  93 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  94 | best_pre= 98.03 | best_post= 98.33 | best_any= 98.33\n",
      "Client  95 | best_pre= 94.83 | best_post= 92.53 | best_any= 94.83\n",
      "Client  96 | best_pre=  9.53 | best_post= 92.60 | best_any= 92.60\n",
      "Client  97 | best_pre= 47.10 | best_post= 97.20 | best_any= 97.20\n",
      "Client  98 | best_pre= 96.03 | best_post= 95.30 | best_any= 96.03\n",
      "Client  99 | best_pre= 42.87 | best_post= 95.83 | best_any= 95.83\n",
      "\n",
      "##### ROUND 5 #####\n",
      "\n",
      "##### ROUND 6 #####\n",
      "\n",
      "##### ROUND 7 #####\n",
      "\n",
      "##### ROUND 8 #####\n",
      "\n",
      "##### ROUND 9 #####\n",
      "\n",
      "##### ROUND 10 #####\n",
      "\n",
      "##### ROUND 11 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.0475\n",
      "avg best PRE   : 76.25%\n",
      "avg best POST  : 86.17%\n",
      "avg best ANY   : 86.67%\n",
      "\n",
      "Client   0 | best_pre= 97.00 | best_post= 94.83 | best_any= 97.00\n",
      "Client   1 | best_pre= 92.63 | best_post= 91.87 | best_any= 92.63\n",
      "Client   2 | best_pre= 55.03 | best_post= 99.20 | best_any= 99.20\n",
      "Client   3 | best_pre= 70.97 | best_post= 98.70 | best_any= 98.70\n",
      "Client   4 | best_pre= 39.30 | best_post= 92.30 | best_any= 92.30\n",
      "Client   5 | best_pre= 97.00 | best_post= 94.90 | best_any= 97.00\n",
      "Client   6 | best_pre= 97.10 | best_post= 96.07 | best_any= 97.10\n",
      "Client   7 | best_pre= 97.17 | best_post= 96.57 | best_any= 97.17\n",
      "Client   8 | best_pre= 39.30 | best_post= 66.40 | best_any= 66.40\n",
      "Client   9 | best_pre= 90.30 | best_post= 94.20 | best_any= 94.20\n",
      "Client  10 | best_pre= 91.30 | best_post= 99.20 | best_any= 99.20\n",
      "Client  11 | best_pre= 40.90 | best_post= 85.03 | best_any= 85.03\n",
      "Client  12 | best_pre= 97.10 | best_post= 98.47 | best_any= 98.47\n",
      "Client  13 | best_pre= 97.17 | best_post= 95.93 | best_any= 97.17\n",
      "Client  14 | best_pre= 73.10 | best_post= 98.57 | best_any= 98.57\n",
      "Client  15 | best_pre= 97.00 | best_post= 96.60 | best_any= 97.00\n",
      "Client  16 | best_pre= 75.13 | best_post= 93.23 | best_any= 93.23\n",
      "Client  17 | best_pre= 96.13 | best_post= 96.10 | best_any= 96.13\n",
      "Client  18 | best_pre= 83.77 | best_post= 99.30 | best_any= 99.30\n",
      "Client  19 | best_pre= 95.60 | best_post= 99.20 | best_any= 99.20\n",
      "Client  20 | best_pre= 97.10 | best_post= 98.70 | best_any= 98.70\n",
      "Client  21 | best_pre= 61.50 | best_post= 94.77 | best_any= 94.77\n",
      "Client  22 | best_pre= 10.13 | best_post= 93.93 | best_any= 93.93\n",
      "Client  23 | best_pre= 93.90 | best_post= 99.00 | best_any= 99.00\n",
      "Client  24 | best_pre= 92.70 | best_post= 93.23 | best_any= 93.23\n",
      "Client  25 | best_pre= 97.10 | best_post= 98.63 | best_any= 98.63\n",
      "Client  26 | best_pre= 97.17 | best_post= 96.27 | best_any= 97.17\n",
      "Client  27 | best_pre= 91.60 | best_post= 97.17 | best_any= 97.17\n",
      "Client  28 | best_pre= 96.03 | best_post= 95.00 | best_any= 96.03\n",
      "Client  29 | best_pre= 90.30 | best_post= 83.10 | best_any= 90.30\n",
      "Client  30 | best_pre= 97.17 | best_post= 96.83 | best_any= 97.17\n",
      "Client  31 | best_pre= 97.17 | best_post= 93.50 | best_any= 97.17\n",
      "Client  32 | best_pre= 55.03 | best_post= 98.50 | best_any= 98.50\n",
      "Client  33 | best_pre= 10.13 | best_post= 84.43 | best_any= 84.43\n",
      "Client  34 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  35 | best_pre= 67.57 | best_post= 98.93 | best_any= 98.93\n",
      "Client  36 | best_pre= 93.63 | best_post= 90.93 | best_any= 93.63\n",
      "Client  37 | best_pre= 96.77 | best_post= 97.03 | best_any= 97.03\n",
      "Client  38 | best_pre= 90.53 | best_post= 86.40 | best_any= 90.53\n",
      "Client  39 | best_pre= 92.00 | best_post= 99.43 | best_any= 99.43\n",
      "Client  40 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  41 | best_pre= 92.63 | best_post= 91.23 | best_any= 92.63\n",
      "Client  42 | best_pre= 97.17 | best_post= 96.33 | best_any= 97.17\n",
      "Client  43 | best_pre= 92.00 | best_post= 99.20 | best_any= 99.20\n",
      "Client  44 | best_pre= 85.93 | best_post= 95.47 | best_any= 95.47\n",
      "Client  45 | best_pre= 93.63 | best_post= 97.07 | best_any= 97.07\n",
      "Client  46 | best_pre= 94.83 | best_post= 95.80 | best_any= 95.80\n",
      "Client  47 | best_pre= 92.70 | best_post= 93.53 | best_any= 93.53\n",
      "Client  48 | best_pre= 97.10 | best_post= 98.50 | best_any= 98.50\n",
      "Client  49 | best_pre= 92.00 | best_post= 99.37 | best_any= 99.37\n",
      "Client  50 | best_pre= 85.93 | best_post= 91.20 | best_any= 91.20\n",
      "Client  51 | best_pre= 65.17 | best_post= 98.47 | best_any= 98.47\n",
      "Client  52 | best_pre= 92.63 | best_post= 92.93 | best_any= 92.93\n",
      "Client  53 | best_pre= 85.57 | best_post= 96.43 | best_any= 96.43\n",
      "Client  54 | best_pre= 76.00 | best_post= 95.00 | best_any= 95.00\n",
      "Client  55 | best_pre= 97.10 | best_post= 99.07 | best_any= 99.07\n",
      "Client  56 | best_pre= 98.03 | best_post= 98.57 | best_any= 98.57\n",
      "Client  57 | best_pre= 97.10 | best_post= 97.83 | best_any= 97.83\n",
      "Client  58 | best_pre= 98.03 | best_post= 97.37 | best_any= 98.03\n",
      "Client  59 | best_pre= 95.90 | best_post= 96.17 | best_any= 96.17\n",
      "Client  60 | best_pre= 93.63 | best_post= 96.97 | best_any= 96.97\n",
      "Client  61 | best_pre= 93.63 | best_post= 95.70 | best_any= 95.70\n",
      "Client  62 | best_pre= 95.00 | best_post= 96.00 | best_any= 96.00\n",
      "Client  63 | best_pre= 93.90 | best_post= 99.60 | best_any= 99.60\n",
      "Client  64 | best_pre= 55.03 | best_post= 98.10 | best_any= 98.10\n",
      "Client  65 | best_pre= 96.13 | best_post= 95.80 | best_any= 96.13\n",
      "Client  66 | best_pre= 91.60 | best_post= 94.57 | best_any= 94.57\n",
      "Client  67 | best_pre= 95.80 | best_post= 95.07 | best_any= 95.80\n",
      "Client  68 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  69 | best_pre= 93.63 | best_post= 93.60 | best_any= 93.63\n",
      "Client  70 | best_pre= 96.13 | best_post= 86.73 | best_any= 96.13\n",
      "Client  71 | best_pre= 97.10 | best_post= 98.93 | best_any= 98.93\n",
      "Client  72 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  73 | best_pre= 40.90 | best_post= 86.97 | best_any= 86.97\n",
      "Client  74 | best_pre= 92.70 | best_post= 92.93 | best_any= 92.93\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  76 | best_pre= 92.70 | best_post= 89.50 | best_any= 92.70\n",
      "Client  77 | best_pre= 95.90 | best_post= 95.40 | best_any= 95.90\n",
      "Client  78 | best_pre= 98.03 | best_post= 95.40 | best_any= 98.03\n",
      "Client  79 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  80 | best_pre= 33.13 | best_post= 44.20 | best_any= 44.20\n",
      "Client  81 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  82 | best_pre= 96.80 | best_post= 96.67 | best_any= 96.80\n",
      "Client  83 | best_pre= 83.77 | best_post= 99.17 | best_any= 99.17\n",
      "Client  84 | best_pre= 52.57 | best_post= 99.47 | best_any= 99.47\n",
      "Client  85 | best_pre= 96.80 | best_post= 96.30 | best_any= 96.80\n",
      "Client  86 | best_pre= 90.53 | best_post= 93.43 | best_any= 93.43\n",
      "Client  87 | best_pre= 95.00 | best_post= 96.80 | best_any= 96.80\n",
      "Client  88 | best_pre= 98.03 | best_post= 98.57 | best_any= 98.57\n",
      "Client  89 | best_pre= 47.10 | best_post= 96.43 | best_any= 96.43\n",
      "Client  90 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  91 | best_pre= 96.77 | best_post= 98.80 | best_any= 98.80\n",
      "Client  92 | best_pre= 91.60 | best_post= 97.07 | best_any= 97.07\n",
      "Client  93 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  94 | best_pre= 98.03 | best_post= 98.33 | best_any= 98.33\n",
      "Client  95 | best_pre= 96.80 | best_post= 95.90 | best_any= 96.80\n",
      "Client  96 | best_pre=  9.53 | best_post= 92.60 | best_any= 92.60\n",
      "Client  97 | best_pre= 47.10 | best_post= 97.20 | best_any= 97.20\n",
      "Client  98 | best_pre= 96.03 | best_post= 96.17 | best_any= 96.17\n",
      "Client  99 | best_pre= 93.63 | best_post= 96.77 | best_any= 96.77\n",
      "\n",
      "##### ROUND 12 #####\n",
      "\n",
      "##### ROUND 13 #####\n",
      "\n",
      "##### ROUND 14 #####\n",
      "\n",
      "##### ROUND 15 #####\n",
      "\n",
      "##### ROUND 16 #####\n",
      "\n",
      "##### ROUND 17 #####\n",
      "\n",
      "##### ROUND 18 #####\n",
      "\n",
      "##### ROUND 19 #####\n",
      "\n",
      "##### ROUND 20 #####\n",
      "\n",
      "##### ROUND 21 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.0292\n",
      "avg best PRE   : 91.97%\n",
      "avg best POST  : 94.60%\n",
      "avg best ANY   : 95.23%\n",
      "\n",
      "Client   0 | best_pre= 97.00 | best_post= 96.00 | best_any= 97.00\n",
      "Client   1 | best_pre= 93.13 | best_post= 95.03 | best_any= 95.03\n",
      "Client   2 | best_pre= 94.67 | best_post= 99.50 | best_any= 99.50\n",
      "Client   3 | best_pre= 70.97 | best_post= 98.87 | best_any= 98.87\n",
      "Client   4 | best_pre= 93.20 | best_post= 96.33 | best_any= 96.33\n",
      "Client   5 | best_pre= 97.00 | best_post= 96.30 | best_any= 97.00\n",
      "Client   6 | best_pre= 97.10 | best_post= 98.23 | best_any= 98.23\n",
      "Client   7 | best_pre= 97.17 | best_post= 96.60 | best_any= 97.17\n",
      "Client   8 | best_pre= 39.30 | best_post= 66.40 | best_any= 66.40\n",
      "Client   9 | best_pre= 94.30 | best_post= 94.20 | best_any= 94.30\n",
      "Client  10 | best_pre= 97.93 | best_post= 99.30 | best_any= 99.30\n",
      "Client  11 | best_pre= 79.47 | best_post= 90.97 | best_any= 90.97\n",
      "Client  12 | best_pre= 97.10 | best_post= 98.90 | best_any= 98.90\n",
      "Client  13 | best_pre= 97.17 | best_post= 96.03 | best_any= 97.17\n",
      "Client  14 | best_pre= 93.73 | best_post= 99.07 | best_any= 99.07\n",
      "Client  15 | best_pre= 97.00 | best_post= 96.60 | best_any= 97.00\n",
      "Client  16 | best_pre= 93.73 | best_post= 93.77 | best_any= 93.77\n",
      "Client  17 | best_pre= 96.13 | best_post= 96.77 | best_any= 96.77\n",
      "Client  18 | best_pre= 83.77 | best_post= 99.30 | best_any= 99.30\n",
      "Client  19 | best_pre= 95.60 | best_post= 99.20 | best_any= 99.20\n",
      "Client  20 | best_pre= 97.10 | best_post= 99.07 | best_any= 99.07\n",
      "Client  21 | best_pre= 91.03 | best_post= 95.70 | best_any= 95.70\n",
      "Client  22 | best_pre= 94.73 | best_post= 95.33 | best_any= 95.33\n",
      "Client  23 | best_pre= 99.03 | best_post= 99.00 | best_any= 99.03\n",
      "Client  24 | best_pre= 93.13 | best_post= 93.83 | best_any= 93.83\n",
      "Client  25 | best_pre= 97.10 | best_post= 98.90 | best_any= 98.90\n",
      "Client  26 | best_pre= 97.17 | best_post= 96.47 | best_any= 97.17\n",
      "Client  27 | best_pre= 93.20 | best_post= 97.63 | best_any= 97.63\n",
      "Client  28 | best_pre= 96.67 | best_post= 95.30 | best_any= 96.67\n",
      "Client  29 | best_pre= 94.73 | best_post= 83.10 | best_any= 94.73\n",
      "Client  30 | best_pre= 97.17 | best_post= 97.20 | best_any= 97.20\n",
      "Client  31 | best_pre= 97.17 | best_post= 93.50 | best_any= 97.17\n",
      "Client  32 | best_pre= 97.93 | best_post= 99.60 | best_any= 99.60\n",
      "Client  33 | best_pre= 94.13 | best_post= 89.70 | best_any= 94.13\n",
      "Client  34 | best_pre= 97.00 | best_post= 99.03 | best_any= 99.03\n",
      "Client  35 | best_pre= 67.57 | best_post= 98.93 | best_any= 98.93\n",
      "Client  36 | best_pre= 93.63 | best_post= 91.37 | best_any= 93.63\n",
      "Client  37 | best_pre= 96.77 | best_post= 97.37 | best_any= 97.37\n",
      "Client  38 | best_pre= 90.53 | best_post= 86.40 | best_any= 90.53\n",
      "Client  39 | best_pre= 99.03 | best_post= 99.47 | best_any= 99.47\n",
      "Client  40 | best_pre= 95.63 | best_post= 94.03 | best_any= 95.63\n",
      "Client  41 | best_pre= 94.73 | best_post= 91.33 | best_any= 94.73\n",
      "Client  42 | best_pre= 97.17 | best_post= 96.57 | best_any= 97.17\n",
      "Client  43 | best_pre= 92.00 | best_post= 99.30 | best_any= 99.30\n",
      "Client  44 | best_pre= 85.93 | best_post= 96.00 | best_any= 96.00\n",
      "Client  45 | best_pre= 93.63 | best_post= 97.17 | best_any= 97.17\n",
      "Client  46 | best_pre= 97.00 | best_post= 96.80 | best_any= 97.00\n",
      "Client  47 | best_pre= 94.13 | best_post= 94.03 | best_any= 94.13\n",
      "Client  48 | best_pre= 97.10 | best_post= 99.20 | best_any= 99.20\n",
      "Client  49 | best_pre= 92.73 | best_post= 99.47 | best_any= 99.47\n",
      "Client  50 | best_pre= 89.27 | best_post= 92.50 | best_any= 92.50\n",
      "Client  51 | best_pre= 65.17 | best_post= 98.53 | best_any= 98.53\n",
      "Client  52 | best_pre= 94.30 | best_post= 94.53 | best_any= 94.53\n",
      "Client  53 | best_pre= 92.97 | best_post= 96.43 | best_any= 96.43\n",
      "Client  54 | best_pre= 92.87 | best_post= 97.03 | best_any= 97.03\n",
      "Client  55 | best_pre= 97.10 | best_post= 99.07 | best_any= 99.07\n",
      "Client  56 | best_pre= 98.03 | best_post= 98.77 | best_any= 98.77\n",
      "Client  57 | best_pre= 97.10 | best_post= 98.80 | best_any= 98.80\n",
      "Client  58 | best_pre= 98.03 | best_post= 97.40 | best_any= 98.03\n",
      "Client  59 | best_pre= 95.90 | best_post= 96.17 | best_any= 96.17\n",
      "Client  60 | best_pre= 93.63 | best_post= 96.97 | best_any= 96.97\n",
      "Client  61 | best_pre= 93.63 | best_post= 97.33 | best_any= 97.33\n",
      "Client  62 | best_pre= 95.00 | best_post= 96.10 | best_any= 96.10\n",
      "Client  63 | best_pre= 97.93 | best_post= 99.60 | best_any= 99.60\n",
      "Client  64 | best_pre= 99.03 | best_post= 99.13 | best_any= 99.13\n",
      "Client  65 | best_pre= 97.17 | best_post= 96.53 | best_any= 97.17\n",
      "Client  66 | best_pre= 91.60 | best_post= 95.50 | best_any= 95.50\n",
      "Client  67 | best_pre= 96.93 | best_post= 96.00 | best_any= 96.93\n",
      "Client  68 | best_pre= 93.57 | best_post= 97.10 | best_any= 97.10\n",
      "Client  69 | best_pre= 93.63 | best_post= 94.23 | best_any= 94.23\n",
      "Client  70 | best_pre= 96.90 | best_post= 86.73 | best_any= 96.90\n",
      "Client  71 | best_pre= 97.10 | best_post= 98.93 | best_any= 98.93\n",
      "Client  72 | best_pre= 93.73 | best_post= 94.33 | best_any= 94.33\n",
      "Client  73 | best_pre= 93.13 | best_post= 88.60 | best_any= 93.13\n",
      "Client  74 | best_pre= 92.70 | best_post= 92.93 | best_any= 92.93\n",
      "Client  75 | best_pre= 99.03 | best_post= 99.60 | best_any= 99.60\n",
      "Client  76 | best_pre= 94.30 | best_post= 90.03 | best_any= 94.30\n",
      "Client  77 | best_pre= 97.00 | best_post= 96.10 | best_any= 97.00\n",
      "Client  78 | best_pre= 98.03 | best_post= 97.37 | best_any= 98.03\n",
      "Client  79 | best_pre= 93.20 | best_post= 96.70 | best_any= 96.70\n",
      "Client  80 | best_pre= 33.13 | best_post= 44.20 | best_any= 44.20\n",
      "Client  81 | best_pre= 97.00 | best_post= 99.60 | best_any= 99.60\n",
      "Client  82 | best_pre= 97.17 | best_post= 96.67 | best_any= 97.17\n",
      "Client  83 | best_pre= 83.77 | best_post= 99.23 | best_any= 99.23\n",
      "Client  84 | best_pre= 92.73 | best_post= 99.60 | best_any= 99.60\n",
      "Client  85 | best_pre= 96.93 | best_post= 96.53 | best_any= 96.93\n",
      "Client  86 | best_pre= 94.13 | best_post= 94.43 | best_any= 94.43\n",
      "Client  87 | best_pre= 95.00 | best_post= 97.50 | best_any= 97.50\n",
      "Client  88 | best_pre= 98.03 | best_post= 98.90 | best_any= 98.90\n",
      "Client  89 | best_pre= 93.73 | best_post= 97.60 | best_any= 97.60\n",
      "Client  90 | best_pre= 97.43 | best_post= 99.67 | best_any= 99.67\n",
      "Client  91 | best_pre= 96.77 | best_post= 99.10 | best_any= 99.10\n",
      "Client  92 | best_pre= 91.60 | best_post= 97.17 | best_any= 97.17\n",
      "Client  93 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  94 | best_pre= 98.03 | best_post= 98.40 | best_any= 98.40\n",
      "Client  95 | best_pre= 97.23 | best_post= 96.93 | best_any= 97.23\n",
      "Client  96 | best_pre= 94.30 | best_post= 94.40 | best_any= 94.40\n",
      "Client  97 | best_pre= 91.03 | best_post= 98.87 | best_any= 98.87\n",
      "Client  98 | best_pre= 97.00 | best_post= 96.17 | best_any= 97.00\n",
      "Client  99 | best_pre= 93.63 | best_post= 97.73 | best_any= 97.73\n",
      "\n",
      "##### ROUND 22 #####\n",
      "\n",
      "##### ROUND 23 #####\n",
      "\n",
      "##### ROUND 24 #####\n",
      "\n",
      "##### ROUND 25 #####\n",
      "\n",
      "##### ROUND 26 #####\n",
      "\n",
      "##### ROUND 27 #####\n",
      "\n",
      "##### ROUND 28 #####\n",
      "\n",
      "##### ROUND 29 #####\n",
      "\n",
      "##### ROUND 30 #####\n",
      "\n",
      "##### ROUND 31 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.0207\n",
      "avg best PRE   : 95.61%\n",
      "avg best POST  : 96.68%\n",
      "avg best ANY   : 97.34%\n",
      "\n",
      "Client   0 | best_pre= 97.00 | best_post= 96.00 | best_any= 97.00\n",
      "Client   1 | best_pre= 94.27 | best_post= 95.13 | best_any= 95.13\n",
      "Client   2 | best_pre= 97.33 | best_post= 99.50 | best_any= 99.50\n",
      "Client   3 | best_pre= 98.53 | best_post= 99.10 | best_any= 99.10\n",
      "Client   4 | best_pre= 94.80 | best_post= 96.70 | best_any= 96.70\n",
      "Client   5 | best_pre= 97.43 | best_post= 96.70 | best_any= 97.43\n",
      "Client   6 | best_pre= 97.10 | best_post= 98.60 | best_any= 98.60\n",
      "Client   7 | best_pre= 97.17 | best_post= 96.60 | best_any= 97.17\n",
      "Client   8 | best_pre= 94.97 | best_post= 88.47 | best_any= 94.97\n",
      "Client   9 | best_pre= 94.30 | best_post= 94.63 | best_any= 94.63\n",
      "Client  10 | best_pre= 97.93 | best_post= 99.30 | best_any= 99.30\n",
      "Client  11 | best_pre= 95.57 | best_post= 93.13 | best_any= 95.57\n",
      "Client  12 | best_pre= 98.13 | best_post= 99.13 | best_any= 99.13\n",
      "Client  13 | best_pre= 97.17 | best_post= 96.03 | best_any= 97.17\n",
      "Client  14 | best_pre= 93.73 | best_post= 99.13 | best_any= 99.13\n",
      "Client  15 | best_pre= 97.00 | best_post= 96.77 | best_any= 97.00\n",
      "Client  16 | best_pre= 93.73 | best_post= 94.43 | best_any= 94.43\n",
      "Client  17 | best_pre= 97.43 | best_post= 97.00 | best_any= 97.43\n",
      "Client  18 | best_pre= 90.20 | best_post= 99.47 | best_any= 99.47\n",
      "Client  19 | best_pre= 95.60 | best_post= 99.43 | best_any= 99.43\n",
      "Client  20 | best_pre= 98.13 | best_post= 99.07 | best_any= 99.07\n",
      "Client  21 | best_pre= 94.03 | best_post= 95.83 | best_any= 95.83\n",
      "Client  22 | best_pre= 94.73 | best_post= 95.33 | best_any= 95.33\n",
      "Client  23 | best_pre= 99.03 | best_post= 99.10 | best_any= 99.10\n",
      "Client  24 | best_pre= 95.57 | best_post= 94.90 | best_any= 95.57\n",
      "Client  25 | best_pre= 97.10 | best_post= 98.90 | best_any= 98.90\n",
      "Client  26 | best_pre= 97.17 | best_post= 96.47 | best_any= 97.17\n",
      "Client  27 | best_pre= 93.20 | best_post= 97.97 | best_any= 97.97\n",
      "Client  28 | best_pre= 97.27 | best_post= 95.30 | best_any= 97.27\n",
      "Client  29 | best_pre= 94.73 | best_post= 83.30 | best_any= 94.73\n",
      "Client  30 | best_pre= 97.20 | best_post= 97.20 | best_any= 97.20\n",
      "Client  31 | best_pre= 97.40 | best_post= 94.80 | best_any= 97.40\n",
      "Client  32 | best_pre= 97.93 | best_post= 99.77 | best_any= 99.77\n",
      "Client  33 | best_pre= 95.57 | best_post= 89.70 | best_any= 95.57\n",
      "Client  34 | best_pre= 97.33 | best_post= 99.03 | best_any= 99.03\n",
      "Client  35 | best_pre= 67.57 | best_post= 99.13 | best_any= 99.13\n",
      "Client  36 | best_pre= 93.63 | best_post= 92.83 | best_any= 93.63\n",
      "Client  37 | best_pre= 98.53 | best_post= 98.03 | best_any= 98.53\n",
      "Client  38 | best_pre= 91.57 | best_post= 88.47 | best_any= 91.57\n",
      "Client  39 | best_pre= 99.03 | best_post= 99.57 | best_any= 99.57\n",
      "Client  40 | best_pre= 95.63 | best_post= 94.03 | best_any= 95.63\n",
      "Client  41 | best_pre= 94.73 | best_post= 92.27 | best_any= 94.73\n",
      "Client  42 | best_pre= 97.27 | best_post= 96.57 | best_any= 97.27\n",
      "Client  43 | best_pre= 97.27 | best_post= 99.30 | best_any= 99.30\n",
      "Client  44 | best_pre= 94.80 | best_post= 96.73 | best_any= 96.73\n",
      "Client  45 | best_pre= 94.80 | best_post= 97.80 | best_any= 97.80\n",
      "Client  46 | best_pre= 97.27 | best_post= 96.80 | best_any= 97.27\n",
      "Client  47 | best_pre= 94.13 | best_post= 94.50 | best_any= 94.50\n",
      "Client  48 | best_pre= 97.10 | best_post= 99.23 | best_any= 99.23\n",
      "Client  49 | best_pre= 92.73 | best_post= 99.47 | best_any= 99.47\n",
      "Client  50 | best_pre= 89.27 | best_post= 92.50 | best_any= 92.50\n",
      "Client  51 | best_pre= 98.53 | best_post= 98.53 | best_any= 98.53\n",
      "Client  52 | best_pre= 94.30 | best_post= 94.53 | best_any= 94.53\n",
      "Client  53 | best_pre= 95.27 | best_post= 96.43 | best_any= 96.43\n",
      "Client  54 | best_pre= 94.97 | best_post= 97.50 | best_any= 97.50\n",
      "Client  55 | best_pre= 98.13 | best_post= 99.07 | best_any= 99.07\n",
      "Client  56 | best_pre= 98.03 | best_post= 99.03 | best_any= 99.03\n",
      "Client  57 | best_pre= 97.10 | best_post= 98.80 | best_any= 98.80\n",
      "Client  58 | best_pre= 98.03 | best_post= 97.47 | best_any= 98.03\n",
      "Client  59 | best_pre= 97.20 | best_post= 96.87 | best_any= 97.20\n",
      "Client  60 | best_pre= 95.27 | best_post= 96.97 | best_any= 96.97\n",
      "Client  61 | best_pre= 94.80 | best_post= 97.40 | best_any= 97.40\n",
      "Client  62 | best_pre= 95.00 | best_post= 97.60 | best_any= 97.60\n",
      "Client  63 | best_pre= 97.93 | best_post= 99.67 | best_any= 99.67\n",
      "Client  64 | best_pre= 99.03 | best_post= 99.13 | best_any= 99.13\n",
      "Client  65 | best_pre= 97.43 | best_post= 96.83 | best_any= 97.43\n",
      "Client  66 | best_pre= 91.60 | best_post= 97.30 | best_any= 97.30\n",
      "Client  67 | best_pre= 97.27 | best_post= 96.33 | best_any= 97.27\n",
      "Client  68 | best_pre= 94.97 | best_post= 97.10 | best_any= 97.10\n",
      "Client  69 | best_pre= 95.27 | best_post= 94.83 | best_any= 95.27\n",
      "Client  70 | best_pre= 97.43 | best_post= 88.33 | best_any= 97.43\n",
      "Client  71 | best_pre= 97.10 | best_post= 98.93 | best_any= 98.93\n",
      "Client  72 | best_pre= 93.73 | best_post= 95.37 | best_any= 95.37\n",
      "Client  73 | best_pre= 93.13 | best_post= 91.47 | best_any= 93.13\n",
      "Client  74 | best_pre= 92.70 | best_post= 93.57 | best_any= 93.57\n",
      "Client  75 | best_pre= 99.03 | best_post= 99.60 | best_any= 99.60\n",
      "Client  76 | best_pre= 94.30 | best_post= 93.23 | best_any= 94.30\n",
      "Client  77 | best_pre= 97.27 | best_post= 96.80 | best_any= 97.27\n",
      "Client  78 | best_pre= 98.03 | best_post= 98.53 | best_any= 98.53\n",
      "Client  79 | best_pre= 94.37 | best_post= 97.37 | best_any= 97.37\n",
      "Client  80 | best_pre= 97.40 | best_post= 94.20 | best_any= 97.40\n",
      "Client  81 | best_pre= 97.33 | best_post= 99.60 | best_any= 99.60\n",
      "Client  82 | best_pre= 97.20 | best_post= 97.13 | best_any= 97.20\n",
      "Client  83 | best_pre= 83.77 | best_post= 99.23 | best_any= 99.23\n",
      "Client  84 | best_pre= 92.73 | best_post= 99.67 | best_any= 99.67\n",
      "Client  85 | best_pre= 97.40 | best_post= 97.13 | best_any= 97.40\n",
      "Client  86 | best_pre= 94.13 | best_post= 94.83 | best_any= 94.83\n",
      "Client  87 | best_pre= 95.00 | best_post= 97.77 | best_any= 97.77\n",
      "Client  88 | best_pre= 98.53 | best_post= 99.07 | best_any= 99.07\n",
      "Client  89 | best_pre= 93.73 | best_post= 97.70 | best_any= 97.70\n",
      "Client  90 | best_pre= 97.43 | best_post= 99.67 | best_any= 99.67\n",
      "Client  91 | best_pre= 96.77 | best_post= 99.17 | best_any= 99.17\n",
      "Client  92 | best_pre= 94.37 | best_post= 97.17 | best_any= 97.17\n",
      "Client  93 | best_pre= 97.33 | best_post= 99.33 | best_any= 99.33\n",
      "Client  94 | best_pre= 98.03 | best_post= 98.53 | best_any= 98.53\n",
      "Client  95 | best_pre= 97.23 | best_post= 96.97 | best_any= 97.23\n",
      "Client  96 | best_pre= 94.30 | best_post= 94.40 | best_any= 94.40\n",
      "Client  97 | best_pre= 98.13 | best_post= 98.87 | best_any= 98.87\n",
      "Client  98 | best_pre= 97.43 | best_post= 96.43 | best_any= 97.43\n",
      "Client  99 | best_pre= 94.23 | best_post= 97.73 | best_any= 97.73\n",
      "\n",
      "##### ROUND 32 #####\n"
     ]
    }
   ],
   "source": [
    "# ============================================================\n",
    "# Clustered-FL primary-update loop\n",
    "# ============================================================\n",
    "#  ADD THIS near the other imports in client_cluster_fl.py\n",
    "\n",
    "n_rounds   = 41\n",
    "print_step = 10              # stats cadence after warm-up\n",
    "loss_buf   = []\n",
    "\n",
    "for rnd in range(n_rounds):\n",
    "\n",
    "    print(f\"\\n##### ROUND {rnd+1} #####\")\n",
    "\n",
    "    # 1) sample users\n",
    "    m            = int(args.frac * args.num_users)\n",
    "    idxs_users   = np.random.choice(args.num_users, m, replace=False)\n",
    "    clusters_now = cluster_participants(idxs_users, clients_clust_id)\n",
    "\n",
    "    # 2) broadcast combined cluster model to each selected user\n",
    "    for uid in idxs_users:\n",
    "        cid = clients_clust_id[uid]\n",
    "        clients[uid].set_state_dict(copy.deepcopy(w_glob_per_cluster[cid]))\n",
    "\n",
    "    # 3) per-cluster FedAvg containers\n",
    "    enc_pool, clf_pool, freq_pool = {}, {}, {}\n",
    "\n",
    "    # 4) local training\n",
    "    for uid in idxs_users:\n",
    "        cid   = clients_clust_id[uid]\n",
    "        ndata = len(net_dataidx_map[uid])\n",
    "\n",
    "        # ---- metrics before train\n",
    "        pre_loss, pre_acc = clients[uid].eval_test()\n",
    "        client_metric[uid][\"best_pre\"]  = max(client_metric[uid][\"best_pre\"],  pre_acc)\n",
    "\n",
    "        # ---- local SGD (own_encoder + classifier only)\n",
    "        loss = clients[uid].train2()\n",
    "        loss_buf.append(loss)\n",
    "\n",
    "        # ---- metrics after train\n",
    "        post_loss, post_acc = clients[uid].eval_test()\n",
    "        client_metric[uid][\"best_post\"] = max(client_metric[uid][\"best_post\"], post_acc)\n",
    "        client_metric[uid][\"best_any\"]  = max(client_metric[uid][\"best_any\"],\n",
    "                                              pre_acc, post_acc)\n",
    "\n",
    "        # ---- stash trainable weights for FedAvg\n",
    "        # Wrap encoder weights with prefix\n",
    "        own_sd = clients[uid].net.own_encoder.state_dict()\n",
    "        own_sd_prefixed = OrderedDict({f\"own_encoder.{k}\": v for k, v in own_sd.items()})\n",
    "        enc_pool.setdefault(cid, []).append(own_sd_prefixed)\n",
    "        \n",
    "        # Wrap classifier weights with prefix\n",
    "        clf_sd = clients[uid].net.classifier.state_dict()\n",
    "        clf_sd_prefixed = OrderedDict({f\"classifier.{k}\": v for k, v in clf_sd.items()})\n",
    "        clf_pool.setdefault(cid, []).append(clf_sd_prefixed)\n",
    "\n",
    "    # 5) FedAvg per cluster (trainable blocks)\n",
    "    for cid in enc_pool:\n",
    "        weights = make_weight_vec(clusters_now[cid], net_dataidx_map)\n",
    "        upd_enc = FedAvg_trainable(enc_pool[cid], weights)\n",
    "        upd_clf = FedAvg_trainable(clf_pool[cid], weights)\n",
    "        w_glob_per_cluster[cid].update(upd_enc)\n",
    "        w_glob_per_cluster[cid].update(upd_clf)\n",
    "    \n",
    "    # ------------------------------------------------------------------\n",
    "    # 6)  Secondary-encoder enrichment  (ONLY for sampled clusters)\n",
    "\n",
    "    local_sec_ep = 10\n",
    "    sec_lr       = 0.01\n",
    "    \n",
    "    for z, clist_sampled in clusters_now.items():\n",
    "    \n",
    "        learners = H_out.get(z, [])\n",
    "        if not learners or len(clist_sampled) == 0:\n",
    "            continue\n",
    "    \n",
    "        # 6.1  Θ2′z  (average current learner weights)\n",
    "        learner_secs = [\n",
    "            {k: v.clone() for k, v in w_glob_per_cluster[j].items()\n",
    "             if k.startswith('secondary_encoder.')}\n",
    "            for j in learners\n",
    "        ]\n",
    "        θ2_prime = fedavg_secondary(learner_secs, [1] * len(learner_secs))\n",
    "    \n",
    "        # strip prefix for bare encoder\n",
    "        base_sec_state = OrderedDict(\n",
    "            (k.replace(\"secondary_encoder.\", \"\"), v) for k, v in θ2_prime.items()\n",
    "        )\n",
    "    \n",
    "        # 6.3  each sampled client returns Δθ_i\n",
    "        delta_list, sizes = [], []\n",
    "        for uid in clist_sampled:\n",
    "            Δθ = clients[uid].train_secondary(base_sec_state,\n",
    "                                              epochs=local_sec_ep,\n",
    "                                              lr=sec_lr)\n",
    "            delta_list.append(Δθ)\n",
    "            sizes.append(len(net_dataidx_map[uid]))\n",
    "    \n",
    "        # 6.4  FedAvg over Δθ_i   →   Δ̄θ(z)\n",
    "        Δ̄θ = fedavg_delta(delta_list, sizes)\n",
    "    \n",
    "        # 6.5  apply Δ̄θ(z) to every learner cluster j\n",
    "        for j in learners:\n",
    "            for k, dv in Δ̄θ.items():                         # k = \"0.weight\", …\n",
    "                key = f\"secondary_encoder.{k}\"               # add prefix\n",
    "                w_glob_per_cluster[j][key] += dv.clone()     # IN-PLACE ADD\n",
    "\n",
    "\n",
    "    # 7) periodic statistics\n",
    "    if rnd < 4:\n",
    "        print_step = 1\n",
    "    else:\n",
    "        print_step = 10\n",
    "\n",
    "    if rnd % print_step == 0:\n",
    "        avg_loss = np.mean(loss_buf) if loss_buf else 0.0\n",
    "        avg_pre  = np.mean([m[\"best_pre\"]  for m in client_metric.values()])\n",
    "        avg_post = np.mean([m[\"best_post\"] for m in client_metric.values()])\n",
    "        avg_any  = np.mean([m[\"best_any\"]  for m in client_metric.values()])\n",
    "\n",
    "        print(\"---- ROUND STATS ----\")\n",
    "        print(f\"avg train loss : {avg_loss:.4f}\")\n",
    "        print(f\"avg best PRE   : {avg_pre :.2f}%\")\n",
    "        print(f\"avg best POST  : {avg_post:.2f}%\")\n",
    "        print(f\"avg best ANY   : {avg_any :.2f}%\\n\")\n",
    "\n",
    "        # optional per-client line (comment if too verbose)\n",
    "        for uid in range(args.num_users):\n",
    "            bm = client_metric[uid]\n",
    "            print(f\"Client {uid:3d} | best_pre={bm['best_pre'] :6.2f} \"\n",
    "                  f\"| best_post={bm['best_post']:6.2f} \"\n",
    "                  f\"| best_any={bm['best_any'] :6.2f}\")\n",
    "\n",
    "        # reset buffers for next report window\n",
    "        loss_buf.clear()\n",
    "        gc.collect()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "print(\"hello\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{0: [1, 2], 1: [4, 0], 2: [1, 4], 3: [4, 0], 4: [2, 1]}\n",
      "Cluster 0, First Client 0: {0: 433, 2: 191, 9: 238}\n",
      "Cluster 1, First Client 1: {2: 616, 6: 115, 8: 18}\n",
      "Cluster 2, First Client 2: {1: 693, 7: 45, 8: 59}\n",
      "Cluster 3, First Client 8: {3: 632, 4: 17, 5: 41}\n",
      "Cluster 4, First Client 10: {4: 265, 6: 215, 7: 240}\n"
     ]
    }
   ],
   "source": [
    "\n",
    "import numpy as np\n",
    "from typing import Dict, List\n",
    "\n",
    "def build_similarity_graph_rank_supply_nonzero(\n",
    "        traindata_cls_counts: Dict[int, Dict[int, int]],\n",
    "        clusters: List[List[int]],\n",
    "        num_classes: int = 10,\n",
    "        top_k: int = 2\n",
    ") -> Dict[int, List[int]]:\n",
    "    \"\"\"\n",
    "    Build a directed Cluster-Complementarity Graph.\n",
    "    Demand is high for rare classes; supply is high for abundant classes.\n",
    "\n",
    "    Parameters\n",
    "    ----------\n",
    "    traindata_cls_counts : {client_id: {class: count}}\n",
    "    clusters             : list of clusters (each is a list of client IDs)\n",
    "    num_classes          : total number of classes\n",
    "    top_k                : keep this many outgoing edges per cluster\n",
    "\n",
    "    Returns\n",
    "    -------\n",
    "    similarity_graph : {cluster_id: [target clusters]}\n",
    "    \"\"\"\n",
    "    C, K = len(clusters), num_classes\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 1. Per-client rank over *non-zero* classes\n",
    "    # ----------------------------------------------------------\n",
    "    client_rank = [{} for _ in range(len(traindata_cls_counts))]\n",
    "    for cid, counts in traindata_cls_counts.items():\n",
    "        present = [(cls, cnt) for cls, cnt in counts.items() if cnt > 0]\n",
    "        if not present:\n",
    "            continue                                # client has no data\n",
    "        present.sort(key=lambda x: x[1])            # ascending (rarest first)\n",
    "        for r, (cls, _) in enumerate(present):\n",
    "            client_rank[cid][cls] = r               # rank 0 … m_i-1\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 2. Demand weight  w[p,k]   (large if class k is rare in cluster p)\n",
    "    # ----------------------------------------------------------\n",
    "    w = np.zeros((C, K), dtype=float)\n",
    "    for p, clist in enumerate(clusters):\n",
    "        for cid in clist:\n",
    "            m_i = len(client_rank[cid])             # number of classes present\n",
    "            for cls, r in client_rank[cid].items():\n",
    "                w[p, cls] += (m_i - r)              # larger when rarer\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 3. Supply strength s[q,k] (large if class k is abundant in cluster q)\n",
    "    # ----------------------------------------------------------\n",
    "    s = np.zeros((C, K), dtype=float)\n",
    "    for q, clist in enumerate(clusters):\n",
    "        for cid in clist:\n",
    "            for cls, r in client_rank[cid].items():\n",
    "                s[q, cls] += (r + 1)                # larger when more common\n",
    "        n_q = max(len(clist), 1)\n",
    "        s[q] /= n_q                                 # per-client normalisation\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 4. Complementarity score matrix  A = W · Sᵀ\n",
    "    # ----------------------------------------------------------\n",
    "    score = w @ s.T\n",
    "    np.fill_diagonal(score, -np.inf)                # forbid self-loops\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 5. Keep top-k outgoing edges for each cluster\n",
    "    # ----------------------------------------------------------\n",
    "    graph = {p: np.argsort(-score[p])[:top_k].tolist() for p in range(C)}\n",
    "    return graph\n",
    "\n",
    "\n",
    "    \n",
    "similarity_graph = build_similarity_graph_rank_supply_nonzero(\n",
    "        traindata_cls_counts,   # your dict\n",
    "        clusters,               # your list of clusters\n",
    "        num_classes=10,\n",
    "        top_k=2\n",
    ")\n",
    "print(similarity_graph)\n",
    "\n",
    "for k in range(len(clusters)):\n",
    "    print(f\"Cluster {k}, First Client {clusters[k][0]}:\", traindata_cls_counts[clusters[k][0]])\n",
    "     "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "metadata": {},
   "outputs": [],
   "source": [
    "def to_model(sd):\n",
    "    mdl = SimpleCNNMNIST2()          # same arch #edit needed, pass args and change based on data\n",
    "    mdl.load_state_dict(sd)\n",
    "    return mdl\n",
    "\n",
    "\n",
    "def FedAvg2(state_dict_list, weight_avg):\n",
    "    \"\"\"\n",
    "    Average a *list* of state_dicts; weight_avg[i] is that model’s weight.\n",
    "    Returns a fresh averaged state_dict.\n",
    "    \"\"\"\n",
    "    averaged = copy.deepcopy(state_dict_list[0])\n",
    "    for key in averaged.keys():\n",
    "        averaged[key] = sum(w * sd[key] for w, sd in zip(weight_avg, state_dict_list))\n",
    "    return averaged\n",
    "\n",
    "def build_secondary_encoder(target_cluster_id: int,\n",
    "                            glob_state_dicts,        # <-- list of OrderedDict\n",
    "                            similarity_graph: dict):\n",
    "    # create target model just to grab its encoder structure\n",
    "    sec_enc = copy.deepcopy(to_model(glob_state_dicts[target_cluster_id]).encoder)\n",
    "    for p in sec_enc.parameters():\n",
    "        p.data.zero_()\n",
    "\n",
    "    # accumulate encoders from similar clusters\n",
    "    for cid in similarity_graph.get(target_cluster_id, []):\n",
    "        enc = to_model(glob_state_dicts[cid]).encoder\n",
    "        for sec_p, src_p in zip(sec_enc.parameters(), enc.parameters()):\n",
    "            sec_p.data += src_p.data\n",
    "    return sec_enc\n",
    "\n",
    "def build_secondary_classifier(\n",
    "        target_cluster_id: int,\n",
    "        glob_state_dicts: List[Dict],        # Phase-1 state-dicts\n",
    "        similarity_graph: Dict[int, List[int]]\n",
    "    ) -> Dict:\n",
    "    \"\"\"\n",
    "    Returns an averaged classifier state_dict (weight & bias) of the\n",
    "    neighbour clusters connected to `target_cluster_id`.\n",
    "    \"\"\"\n",
    "    neighbours = similarity_graph.get(target_cluster_id, [])\n",
    "    if not neighbours:                       # no neighbours ⇒ zero init\n",
    "        # weight: (10,84) ; bias: (10,)\n",
    "        return {\n",
    "            'weight': torch.zeros_like(\n",
    "                glob_state_dicts[target_cluster_id]['classifier.weight']),\n",
    "            'bias':   torch.zeros_like(\n",
    "                glob_state_dicts[target_cluster_id]['classifier.bias']),\n",
    "        }\n",
    "\n",
    "    # ---- accumulate weights / biases ----\n",
    "    w_sum = 0.0\n",
    "    b_sum = 0.0\n",
    "    for cid in neighbours:\n",
    "        w_sum += glob_state_dicts[cid]['classifier.weight']\n",
    "        b_sum += glob_state_dicts[cid]['classifier.bias']\n",
    "    w_avg = w_sum / len(neighbours)\n",
    "    b_avg = b_sum / len(neighbours)\n",
    "\n",
    "    return {'weight': w_avg.clone(), 'bias': b_avg.clone()}\n",
    "\n",
    "# --- Assume you already ran Phase-1 & have: ----------------------------\n",
    "# clusters               : List[List[int]] – client indices per cluster\n",
    "# global_cluster_models  : Dict[int, SimpleCNN] – best model per cluster\n",
    "# similarity_graph       : Dict[int, List[int]] – your similarity edges\n",
    "# clients                : List[Client_ClusterFL] (place-holders for now)\n",
    "# -----------------------------------------------------------------------\n",
    "\n",
    "# ------ Build CombinedModel and push to each client  -------------------\n",
    "### >>> MODIFIED LOOP STARTS HERE <<<\n",
    "global_cluster_models = copy.deepcopy(w_glob_per_cluster)   # Phase-1 best SDs\n",
    "\n",
    "for cid, client_idxs in enumerate(clusters):\n",
    "    # ---- rebuild Phase-1 model once -------------------------\n",
    "    own_model = to_model(global_cluster_models[cid])         # SimpleCNNMNIST2()\n",
    "    own_enc   = own_model.encoder\n",
    "    own_clf_sd = own_model.classifier.state_dict()\n",
    "\n",
    "    # ---- secondary encoder ---------------------------------\n",
    "    sec_enc = build_secondary_encoder(\n",
    "        cid,\n",
    "        global_cluster_models,\n",
    "        similarity_graph\n",
    "    )\n",
    "\n",
    "    # ---- secondary classifier ------------------------------\n",
    "    sec_clf_sd = build_secondary_classifier(\n",
    "        cid,\n",
    "        global_cluster_models,\n",
    "        similarity_graph\n",
    "    )\n",
    "\n",
    "    # ---- assign CombinedModel to every client --------------\n",
    "    for idx in client_idxs:\n",
    "        clients[idx].net = CombinedModelMNIST(\n",
    "            own_encoder       = own_enc,\n",
    "            secondary_encoder = sec_enc,\n",
    "            own_clf_sd        = own_clf_sd,\n",
    "            sec_clf_sd        = sec_clf_sd,\n",
    "            num_classes       = 10\n",
    "        )\n",
    "### <<< MODIFIED LOOP ENDS HERE >>>"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "###### P2-ROUND 1 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0192\n",
      "Init Test Acc:  76.23 Final Test Acc: 94.18\n",
      "Client   0  cur=92.17  best=92.17\n",
      "Client   1  cur=52.00  best= 0.00\n",
      "Client   2  cur=93.80  best=93.80\n",
      "Client   3  cur=52.00  best= 0.00\n",
      "Client   4  cur=85.60  best= 0.00\n",
      "Client   5  cur=95.70  best=95.70\n",
      "Client   6  cur=82.23  best= 0.00\n",
      "Client   7  cur=85.60  best= 0.00\n",
      "Client   8  cur=85.60  best= 0.00\n",
      "Client   9  cur=62.27  best= 0.00\n",
      "Client  10  cur=62.27  best= 0.00\n",
      "Client  11  cur=82.23  best= 0.00\n",
      "Client  12  cur=52.00  best= 0.00\n",
      "Client  13  cur=85.60  best= 0.00\n",
      "Client  14  cur=62.27  best= 0.00\n",
      "Client  15  cur=85.67  best= 0.00\n",
      "Client  16  cur=62.27  best= 0.00\n",
      "Client  17  cur=85.60  best= 0.00\n",
      "Client  18  cur=82.23  best= 0.00\n",
      "Client  19  cur=52.00  best= 0.00\n",
      "Client  20  cur=62.27  best= 0.00\n",
      "Client  21  cur=82.23  best= 0.00\n",
      "Client  22  cur=52.00  best= 0.00\n",
      "Client  23  cur=85.67  best= 0.00\n",
      "Client  24  cur=85.60  best= 0.00\n",
      "Client  25  cur=95.67  best=95.67\n",
      "Client  26  cur=52.00  best= 0.00\n",
      "Client  27  cur=62.27  best= 0.00\n",
      "Client  28  cur=82.23  best= 0.00\n",
      "Client  29  cur=96.87  best=96.87\n",
      "Client  30  cur=85.67  best= 0.00\n",
      "Client  31  cur=85.67  best= 0.00\n",
      "Client  32  cur=82.23  best= 0.00\n",
      "Client  33  cur=95.20  best=95.20\n",
      "Client  34  cur=52.00  best= 0.00\n",
      "Client  35  cur=98.13  best=98.13\n",
      "Client  36  cur=82.23  best= 0.00\n",
      "Client  37  cur=82.23  best= 0.00\n",
      "Client  38  cur=85.60  best= 0.00\n",
      "Client  39  cur=62.27  best= 0.00\n",
      "Client  40  cur=52.00  best= 0.00\n",
      "Client  41  cur=62.27  best= 0.00\n",
      "Client  42  cur=62.27  best= 0.00\n",
      "Client  43  cur=62.27  best= 0.00\n",
      "Client  44  cur=97.50  best=97.50\n",
      "Client  45  cur=52.00  best= 0.00\n",
      "Client  46  cur=91.57  best=91.57\n",
      "Client  47  cur=85.60  best= 0.00\n",
      "Client  48  cur=62.27  best= 0.00\n",
      "Client  49  cur=52.00  best= 0.00\n",
      "Client  50  cur=95.90  best=95.90\n",
      "Client  51  cur=85.67  best= 0.00\n",
      "Client  52  cur=85.67  best= 0.00\n",
      "Client  53  cur=72.73  best=72.73\n",
      "Client  54  cur=94.33  best=94.33\n",
      "Client  55  cur=62.27  best= 0.00\n",
      "Client  56  cur=82.23  best= 0.00\n",
      "Client  57  cur=82.23  best= 0.00\n",
      "Client  58  cur=85.60  best= 0.00\n",
      "Client  59  cur=62.27  best= 0.00\n",
      "Client  60  cur=85.67  best= 0.00\n",
      "Client  61  cur=99.07  best=99.07\n",
      "Client  62  cur=82.23  best= 0.00\n",
      "Client  63  cur=52.00  best= 0.00\n",
      "Client  64  cur=95.33  best=95.33\n",
      "Client  65  cur=85.67  best= 0.00\n",
      "Client  66  cur=85.60  best= 0.00\n",
      "Client  67  cur=96.57  best=96.57\n",
      "Client  68  cur=85.60  best= 0.00\n",
      "Client  69  cur=62.27  best= 0.00\n",
      "Client  70  cur=82.23  best= 0.00\n",
      "Client  71  cur=62.27  best= 0.00\n",
      "Client  72  cur=52.00  best= 0.00\n",
      "Client  73  cur=98.67  best=98.67\n",
      "Client  74  cur=85.67  best= 0.00\n",
      "Client  75  cur=62.27  best= 0.00\n",
      "Client  76  cur=85.67  best= 0.00\n",
      "Client  77  cur=82.23  best= 0.00\n",
      "Client  78  cur=62.27  best= 0.00\n",
      "Client  79  cur=98.83  best=98.83\n",
      "Client  80  cur=90.70  best=90.70\n",
      "Client  81  cur=90.87  best=90.87\n",
      "Client  82  cur=85.60  best= 0.00\n",
      "Client  83  cur=85.60  best= 0.00\n",
      "Client  84  cur=85.67  best= 0.00\n",
      "Client  85  cur=62.27  best= 0.00\n",
      "Client  86  cur=85.67  best= 0.00\n",
      "Client  87  cur=85.60  best= 0.00\n",
      "Client  88  cur=62.27  best= 0.00\n",
      "Client  89  cur=85.67  best= 0.00\n",
      "Client  90  cur=85.60  best= 0.00\n",
      "Client  91  cur=52.00  best= 0.00\n",
      "Client  92  cur=82.23  best= 0.00\n",
      "Client  93  cur=52.00  best= 0.00\n",
      "Client  94  cur=85.60  best= 0.00\n",
      "Client  95  cur=85.60  best= 0.00\n",
      "Client  96  cur=85.60  best= 0.00\n",
      "Client  97  cur=85.60  best= 0.00\n",
      "Client  98  cur=94.07  best=94.07\n",
      "Client  99  cur=82.23  best= 0.00\n",
      "Round 1 AvgCur=77.68  AvgBest=18.84\n",
      "\n",
      "###### P2-ROUND 2 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0159\n",
      "Init Test Acc:  96.38 Final Test Acc: 95.35\n",
      "Client   0  cur=92.17  best=92.17\n",
      "Client   1  cur=52.00  best= 0.00\n",
      "Client   2  cur=93.80  best=93.80\n",
      "Client   3  cur=98.43  best=98.97\n",
      "Client   4  cur=85.60  best= 0.00\n",
      "Client   5  cur=95.70  best=95.70\n",
      "Client   6  cur=82.23  best= 0.00\n",
      "Client   7  cur=85.60  best= 0.00\n",
      "Client   8  cur=95.27  best=95.83\n",
      "Client   9  cur=62.27  best= 0.00\n",
      "Client  10  cur=62.27  best= 0.00\n",
      "Client  11  cur=82.23  best= 0.00\n",
      "Client  12  cur=97.73  best=98.97\n",
      "Client  13  cur=93.30  best=95.83\n",
      "Client  14  cur=62.27  best= 0.00\n",
      "Client  15  cur=93.17  best=96.00\n",
      "Client  16  cur=62.27  best= 0.00\n",
      "Client  17  cur=94.93  best=95.83\n",
      "Client  18  cur=82.23  best= 0.00\n",
      "Client  19  cur=98.97  best=98.97\n",
      "Client  20  cur=62.27  best= 0.00\n",
      "Client  21  cur=82.23  best= 0.00\n",
      "Client  22  cur=52.00  best= 0.00\n",
      "Client  23  cur=85.67  best= 0.00\n",
      "Client  24  cur=85.60  best= 0.00\n",
      "Client  25  cur=95.67  best=95.67\n",
      "Client  26  cur=52.00  best= 0.00\n",
      "Client  27  cur=62.27  best= 0.00\n",
      "Client  28  cur=82.23  best= 0.00\n",
      "Client  29  cur=96.87  best=96.87\n",
      "Client  30  cur=95.87  best=96.00\n",
      "Client  31  cur=85.67  best= 0.00\n",
      "Client  32  cur=82.23  best= 0.00\n",
      "Client  33  cur=95.20  best=95.20\n",
      "Client  34  cur=98.97  best=98.97\n",
      "Client  35  cur=98.13  best=98.13\n",
      "Client  36  cur=98.47  best=98.47\n",
      "Client  37  cur=97.30  best=98.23\n",
      "Client  38  cur=85.60  best= 0.00\n",
      "Client  39  cur=62.27  best= 0.00\n",
      "Client  40  cur=52.00  best= 0.00\n",
      "Client  41  cur=92.13  best=92.13\n",
      "Client  42  cur=62.27  best= 0.00\n",
      "Client  43  cur=92.87  best=92.87\n",
      "Client  44  cur=97.50  best=97.50\n",
      "Client  45  cur=52.00  best= 0.00\n",
      "Client  46  cur=91.57  best=91.57\n",
      "Client  47  cur=85.60  best= 0.00\n",
      "Client  48  cur=91.67  best=91.70\n",
      "Client  49  cur=52.00  best= 0.00\n",
      "Client  50  cur=95.90  best=95.90\n",
      "Client  51  cur=85.67  best= 0.00\n",
      "Client  52  cur=85.67  best= 0.00\n",
      "Client  53  cur=72.73  best=72.73\n",
      "Client  54  cur=93.37  best=95.83\n",
      "Client  55  cur=62.27  best= 0.00\n",
      "Client  56  cur=82.23  best= 0.00\n",
      "Client  57  cur=98.03  best=98.23\n",
      "Client  58  cur=85.60  best= 0.00\n",
      "Client  59  cur=62.27  best= 0.00\n",
      "Client  60  cur=94.83  best=96.00\n",
      "Client  61  cur=99.07  best=99.07\n",
      "Client  62  cur=82.23  best= 0.00\n",
      "Client  63  cur=52.00  best= 0.00\n",
      "Client  64  cur=95.33  best=95.33\n",
      "Client  65  cur=85.67  best= 0.00\n",
      "Client  66  cur=85.60  best= 0.00\n",
      "Client  67  cur=96.57  best=96.57\n",
      "Client  68  cur=85.60  best= 0.00\n",
      "Client  69  cur=62.27  best= 0.00\n",
      "Client  70  cur=82.23  best= 0.00\n",
      "Client  71  cur=62.27  best= 0.00\n",
      "Client  72  cur=52.00  best= 0.00\n",
      "Client  73  cur=98.67  best=98.67\n",
      "Client  74  cur=85.67  best= 0.00\n",
      "Client  75  cur=62.27  best= 0.00\n",
      "Client  76  cur=85.67  best= 0.00\n",
      "Client  77  cur=82.23  best= 0.00\n",
      "Client  78  cur=62.27  best= 0.00\n",
      "Client  79  cur=98.97  best=98.97\n",
      "Client  80  cur=90.70  best=90.70\n",
      "Client  81  cur=90.87  best=90.87\n",
      "Client  82  cur=85.60  best= 0.00\n",
      "Client  83  cur=85.60  best= 0.00\n",
      "Client  84  cur=85.67  best= 0.00\n",
      "Client  85  cur=62.27  best= 0.00\n",
      "Client  86  cur=85.67  best= 0.00\n",
      "Client  87  cur=95.73  best=95.83\n",
      "Client  88  cur=62.27  best= 0.00\n",
      "Client  89  cur=85.67  best= 0.00\n",
      "Client  90  cur=85.60  best= 0.00\n",
      "Client  91  cur=52.00  best= 0.00\n",
      "Client  92  cur=82.23  best= 0.00\n",
      "Client  93  cur=52.00  best= 0.00\n",
      "Client  94  cur=85.60  best= 0.00\n",
      "Client  95  cur=85.60  best= 0.00\n",
      "Client  96  cur=87.07  best=95.83\n",
      "Client  97  cur=85.60  best= 0.00\n",
      "Client  98  cur=94.07  best=94.07\n",
      "Client  99  cur=82.23  best= 0.00\n",
      "Round 2 AvgCur=81.56  AvgBest=36.20\n",
      "\n",
      "###### P2-ROUND 3 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0140\n",
      "Init Test Acc:  96.17 Final Test Acc: 94.87\n",
      "Client   0  cur=93.30  best=95.77\n",
      "Client   1  cur=52.00  best= 0.00\n",
      "Client   2  cur=93.80  best=93.80\n",
      "Client   3  cur=98.43  best=98.97\n",
      "Client   4  cur=85.60  best= 0.00\n",
      "Client   5  cur=95.70  best=95.70\n",
      "Client   6  cur=82.23  best= 0.00\n",
      "Client   7  cur=95.53  best=95.77\n",
      "Client   8  cur=95.27  best=95.83\n",
      "Client   9  cur=62.27  best= 0.00\n",
      "Client  10  cur=62.27  best= 0.00\n",
      "Client  11  cur=82.23  best= 0.00\n",
      "Client  12  cur=97.73  best=98.97\n",
      "Client  13  cur=93.30  best=95.83\n",
      "Client  14  cur=62.27  best= 0.00\n",
      "Client  15  cur=93.17  best=96.00\n",
      "Client  16  cur=92.70  best=93.00\n",
      "Client  17  cur=94.93  best=95.83\n",
      "Client  18  cur=82.23  best= 0.00\n",
      "Client  19  cur=98.97  best=98.97\n",
      "Client  20  cur=62.27  best= 0.00\n",
      "Client  21  cur=82.23  best= 0.00\n",
      "Client  22  cur=52.00  best= 0.00\n",
      "Client  23  cur=85.67  best= 0.00\n",
      "Client  24  cur=94.33  best=95.77\n",
      "Client  25  cur=95.67  best=95.67\n",
      "Client  26  cur=52.00  best= 0.00\n",
      "Client  27  cur=62.27  best= 0.00\n",
      "Client  28  cur=82.23  best= 0.00\n",
      "Client  29  cur=96.87  best=97.73\n",
      "Client  30  cur=95.87  best=96.00\n",
      "Client  31  cur=85.67  best= 0.00\n",
      "Client  32  cur=82.23  best= 0.00\n",
      "Client  33  cur=95.47  best=95.47\n",
      "Client  34  cur=98.97  best=98.97\n",
      "Client  35  cur=98.13  best=98.13\n",
      "Client  36  cur=98.47  best=98.47\n",
      "Client  37  cur=97.23  best=98.23\n",
      "Client  38  cur=85.60  best= 0.00\n",
      "Client  39  cur=62.27  best= 0.00\n",
      "Client  40  cur=52.00  best= 0.00\n",
      "Client  41  cur=92.13  best=92.13\n",
      "Client  42  cur=62.27  best= 0.00\n",
      "Client  43  cur=92.43  best=93.00\n",
      "Client  44  cur=97.50  best=97.50\n",
      "Client  45  cur=52.00  best= 0.00\n",
      "Client  46  cur=91.63  best=95.77\n",
      "Client  47  cur=85.60  best= 0.00\n",
      "Client  48  cur=91.67  best=91.70\n",
      "Client  49  cur=52.00  best= 0.00\n",
      "Client  50  cur=95.90  best=95.90\n",
      "Client  51  cur=95.10  best=95.33\n",
      "Client  52  cur=85.67  best= 0.00\n",
      "Client  53  cur=76.57  best=93.00\n",
      "Client  54  cur=93.37  best=95.83\n",
      "Client  55  cur=62.27  best= 0.00\n",
      "Client  56  cur=98.07  best=98.07\n",
      "Client  57  cur=98.03  best=98.23\n",
      "Client  58  cur=85.60  best= 0.00\n",
      "Client  59  cur=92.67  best=93.00\n",
      "Client  60  cur=94.83  best=96.00\n",
      "Client  61  cur=99.03  best=99.07\n",
      "Client  62  cur=82.23  best= 0.00\n",
      "Client  63  cur=52.00  best= 0.00\n",
      "Client  64  cur=95.70  best=95.70\n",
      "Client  65  cur=85.67  best= 0.00\n",
      "Client  66  cur=85.60  best= 0.00\n",
      "Client  67  cur=96.57  best=96.57\n",
      "Client  68  cur=85.60  best= 0.00\n",
      "Client  69  cur=62.27  best= 0.00\n",
      "Client  70  cur=82.23  best= 0.00\n",
      "Client  71  cur=62.27  best= 0.00\n",
      "Client  72  cur=99.10  best=99.10\n",
      "Client  73  cur=98.90  best=99.00\n",
      "Client  74  cur=96.33  best=96.33\n",
      "Client  75  cur=62.27  best= 0.00\n",
      "Client  76  cur=85.67  best= 0.00\n",
      "Client  77  cur=82.23  best= 0.00\n",
      "Client  78  cur=62.27  best= 0.00\n",
      "Client  79  cur=99.00  best=99.00\n",
      "Client  80  cur=90.70  best=90.70\n",
      "Client  81  cur=90.87  best=90.87\n",
      "Client  82  cur=85.60  best= 0.00\n",
      "Client  83  cur=85.60  best= 0.00\n",
      "Client  84  cur=85.67  best= 0.00\n",
      "Client  85  cur=62.27  best= 0.00\n",
      "Client  86  cur=85.67  best= 0.00\n",
      "Client  87  cur=95.73  best=95.83\n",
      "Client  88  cur=62.27  best= 0.00\n",
      "Client  89  cur=85.67  best= 0.00\n",
      "Client  90  cur=85.60  best= 0.00\n",
      "Client  91  cur=52.00  best= 0.00\n",
      "Client  92  cur=82.23  best= 0.00\n",
      "Client  93  cur=52.00  best= 0.00\n",
      "Client  94  cur=85.60  best= 0.00\n",
      "Client  95  cur=85.60  best= 0.00\n",
      "Client  96  cur=87.07  best=95.83\n",
      "Client  97  cur=85.60  best= 0.00\n",
      "Client  98  cur=94.07  best=94.07\n",
      "Client  99  cur=97.47  best=97.73\n",
      "Round 3 AvgCur=83.39  AvgBest=45.14\n",
      "\n",
      "###### P2-ROUND 4 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0154\n",
      "Init Test Acc:  95.78 Final Test Acc: 94.26\n",
      "Client   0  cur=93.30  best=95.77\n",
      "Client   1  cur=52.00  best= 0.00\n",
      "Client   2  cur=93.80  best=93.80\n",
      "Client   3  cur=98.43  best=98.97\n",
      "Client   4  cur=85.60  best= 0.00\n",
      "Client   5  cur=95.70  best=95.70\n",
      "Client   6  cur=82.23  best= 0.00\n",
      "Client   7  cur=95.53  best=95.77\n",
      "Client   8  cur=95.27  best=95.83\n",
      "Client   9  cur=62.27  best= 0.00\n",
      "Client  10  cur=62.27  best= 0.00\n",
      "Client  11  cur=82.23  best= 0.00\n",
      "Client  12  cur=98.53  best=99.00\n",
      "Client  13  cur=93.30  best=95.83\n",
      "Client  14  cur=62.27  best= 0.00\n",
      "Client  15  cur=93.17  best=96.00\n",
      "Client  16  cur=92.70  best=93.00\n",
      "Client  17  cur=94.93  best=95.83\n",
      "Client  18  cur=98.50  best=98.50\n",
      "Client  19  cur=98.97  best=98.97\n",
      "Client  20  cur=92.07  best=92.53\n",
      "Client  21  cur=97.83  best=97.87\n",
      "Client  22  cur=52.00  best= 0.00\n",
      "Client  23  cur=85.67  best= 0.00\n",
      "Client  24  cur=94.43  best=96.10\n",
      "Client  25  cur=95.67  best=95.67\n",
      "Client  26  cur=52.00  best= 0.00\n",
      "Client  27  cur=62.27  best= 0.00\n",
      "Client  28  cur=82.23  best= 0.00\n",
      "Client  29  cur=96.77  best=97.87\n",
      "Client  30  cur=95.87  best=96.00\n",
      "Client  31  cur=85.67  best= 0.00\n",
      "Client  32  cur=82.23  best= 0.00\n",
      "Client  33  cur=95.47  best=95.47\n",
      "Client  34  cur=98.97  best=98.97\n",
      "Client  35  cur=98.13  best=98.13\n",
      "Client  36  cur=98.33  best=98.47\n",
      "Client  37  cur=97.23  best=98.23\n",
      "Client  38  cur=85.60  best= 0.00\n",
      "Client  39  cur=92.60  best=92.60\n",
      "Client  40  cur=52.00  best= 0.00\n",
      "Client  41  cur=92.13  best=92.13\n",
      "Client  42  cur=86.13  best=92.53\n",
      "Client  43  cur=92.43  best=93.00\n",
      "Client  44  cur=97.50  best=97.50\n",
      "Client  45  cur=52.00  best= 0.00\n",
      "Client  46  cur=91.63  best=95.77\n",
      "Client  47  cur=87.57  best=96.10\n",
      "Client  48  cur=91.67  best=91.70\n",
      "Client  49  cur=52.00  best= 0.00\n",
      "Client  50  cur=95.90  best=95.90\n",
      "Client  51  cur=94.97  best=96.60\n",
      "Client  52  cur=95.23  best=96.60\n",
      "Client  53  cur=76.57  best=93.00\n",
      "Client  54  cur=93.37  best=95.83\n",
      "Client  55  cur=62.27  best= 0.00\n",
      "Client  56  cur=98.07  best=98.07\n",
      "Client  57  cur=98.03  best=98.23\n",
      "Client  58  cur=85.60  best= 0.00\n",
      "Client  59  cur=92.67  best=93.00\n",
      "Client  60  cur=94.33  best=96.60\n",
      "Client  61  cur=99.03  best=99.07\n",
      "Client  62  cur=82.23  best= 0.00\n",
      "Client  63  cur=52.00  best= 0.00\n",
      "Client  64  cur=95.70  best=95.70\n",
      "Client  65  cur=85.67  best= 0.00\n",
      "Client  66  cur=85.60  best= 0.00\n",
      "Client  67  cur=96.90  best=97.87\n",
      "Client  68  cur=95.70  best=96.10\n",
      "Client  69  cur=62.27  best= 0.00\n",
      "Client  70  cur=82.23  best= 0.00\n",
      "Client  71  cur=62.27  best= 0.00\n",
      "Client  72  cur=99.10  best=99.10\n",
      "Client  73  cur=98.90  best=99.00\n",
      "Client  74  cur=96.33  best=96.33\n",
      "Client  75  cur=62.27  best= 0.00\n",
      "Client  76  cur=85.67  best= 0.00\n",
      "Client  77  cur=82.23  best= 0.00\n",
      "Client  78  cur=92.30  best=92.53\n",
      "Client  79  cur=99.00  best=99.00\n",
      "Client  80  cur=90.70  best=90.70\n",
      "Client  81  cur=90.87  best=90.87\n",
      "Client  82  cur=85.60  best= 0.00\n",
      "Client  83  cur=85.60  best= 0.00\n",
      "Client  84  cur=85.67  best= 0.00\n",
      "Client  85  cur=86.03  best=92.53\n",
      "Client  86  cur=85.67  best= 0.00\n",
      "Client  87  cur=95.73  best=95.83\n",
      "Client  88  cur=93.27  best=93.27\n",
      "Client  89  cur=85.67  best= 0.00\n",
      "Client  90  cur=85.60  best= 0.00\n",
      "Client  91  cur=52.00  best= 0.00\n",
      "Client  92  cur=98.13  best=98.13\n",
      "Client  93  cur=52.00  best= 0.00\n",
      "Client  94  cur=95.60  best=96.10\n",
      "Client  95  cur=85.60  best= 0.00\n",
      "Client  96  cur=87.07  best=95.83\n",
      "Client  97  cur=85.60  best= 0.00\n",
      "Client  98  cur=94.07  best=94.07\n",
      "Client  99  cur=97.47  best=97.73\n",
      "Round 4 AvgCur=85.87  AvgBest=57.53\n",
      "\n",
      "###### P2-ROUND 5 ######\n",
      "###### P2-ROUND 6 ######\n",
      "###### P2-ROUND 7 ######\n",
      "###### P2-ROUND 8 ######\n",
      "###### P2-ROUND 9 ######\n",
      "###### P2-ROUND 10 ######\n",
      "###### P2-ROUND 11 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0083\n",
      "Init Test Acc:  96.82 Final Test Acc: 95.15\n",
      "Client   0  cur=93.37  best=95.77\n",
      "Client   1  cur=52.00  best= 0.00\n",
      "Client   2  cur=94.33  best=98.17\n",
      "Client   3  cur=98.93  best=98.97\n",
      "Client   4  cur=95.67  best=95.73\n",
      "Client   5  cur=95.73  best=96.20\n",
      "Client   6  cur=82.23  best= 0.00\n",
      "Client   7  cur=95.57  best=95.77\n",
      "Client   8  cur=95.27  best=95.83\n",
      "Client   9  cur=92.03  best=92.03\n",
      "Client  10  cur=92.90  best=92.90\n",
      "Client  11  cur=97.37  best=98.17\n",
      "Client  12  cur=98.17  best=99.00\n",
      "Client  13  cur=92.73  best=96.20\n",
      "Client  14  cur=93.07  best=93.33\n",
      "Client  15  cur=92.93  best=96.00\n",
      "Client  16  cur=92.70  best=93.00\n",
      "Client  17  cur=95.27  best=95.83\n",
      "Client  18  cur=98.57  best=98.57\n",
      "Client  19  cur=99.07  best=99.07\n",
      "Client  20  cur=92.07  best=92.53\n",
      "Client  21  cur=98.10  best=98.30\n",
      "Client  22  cur=99.13  best=99.13\n",
      "Client  23  cur=94.30  best=94.30\n",
      "Client  24  cur=94.67  best=96.10\n",
      "Client  25  cur=95.83  best=95.83\n",
      "Client  26  cur=99.23  best=99.23\n",
      "Client  27  cur=62.27  best= 0.00\n",
      "Client  28  cur=98.17  best=98.17\n",
      "Client  29  cur=96.77  best=97.87\n",
      "Client  30  cur=95.87  best=96.00\n",
      "Client  31  cur=96.53  best=96.53\n",
      "Client  32  cur=97.97  best=98.30\n",
      "Client  33  cur=95.33  best=96.93\n",
      "Client  34  cur=99.00  best=99.17\n",
      "Client  35  cur=98.43  best=98.43\n",
      "Client  36  cur=98.37  best=98.47\n",
      "Client  37  cur=97.23  best=98.23\n",
      "Client  38  cur=95.73  best=95.97\n",
      "Client  39  cur=93.13  best=93.13\n",
      "Client  40  cur=99.03  best=99.17\n",
      "Client  41  cur=92.43  best=93.73\n",
      "Client  42  cur=86.13  best=92.53\n",
      "Client  43  cur=92.43  best=93.00\n",
      "Client  44  cur=97.80  best=97.80\n",
      "Client  45  cur=99.23  best=99.23\n",
      "Client  46  cur=92.07  best=95.77\n",
      "Client  47  cur=86.77  best=96.20\n",
      "Client  48  cur=92.80  best=92.80\n",
      "Client  49  cur=98.73  best=99.17\n",
      "Client  50  cur=95.83  best=95.90\n",
      "Client  51  cur=95.43  best=96.60\n",
      "Client  52  cur=95.67  best=96.60\n",
      "Client  53  cur=77.13  best=93.73\n",
      "Client  54  cur=93.17  best=95.83\n",
      "Client  55  cur=92.50  best=92.50\n",
      "Client  56  cur=98.07  best=98.07\n",
      "Client  57  cur=97.97  best=98.23\n",
      "Client  58  cur=95.70  best=95.70\n",
      "Client  59  cur=92.67  best=93.00\n",
      "Client  60  cur=95.13  best=96.93\n",
      "Client  61  cur=99.37  best=99.37\n",
      "Client  62  cur=98.17  best=98.17\n",
      "Client  63  cur=52.00  best= 0.00\n",
      "Client  64  cur=95.67  best=96.60\n",
      "Client  65  cur=94.97  best=94.97\n",
      "Client  66  cur=90.40  best=96.20\n",
      "Client  67  cur=97.00  best=98.17\n",
      "Client  68  cur=95.90  best=96.10\n",
      "Client  69  cur=79.03  best=79.03\n",
      "Client  70  cur=97.53  best=97.53\n",
      "Client  71  cur=91.27  best=93.73\n",
      "Client  72  cur=99.07  best=99.10\n",
      "Client  73  cur=99.13  best=99.13\n",
      "Client  74  cur=96.13  best=96.33\n",
      "Client  75  cur=62.27  best= 0.00\n",
      "Client  76  cur=96.00  best=96.00\n",
      "Client  77  cur=98.43  best=98.43\n",
      "Client  78  cur=92.30  best=92.53\n",
      "Client  79  cur=99.20  best=99.20\n",
      "Client  80  cur=91.33  best=91.33\n",
      "Client  81  cur=90.87  best=90.87\n",
      "Client  82  cur=93.10  best=93.10\n",
      "Client  83  cur=93.60  best=94.03\n",
      "Client  84  cur=95.83  best=96.93\n",
      "Client  85  cur=85.53  best=92.53\n",
      "Client  86  cur=96.07  best=96.93\n",
      "Client  87  cur=95.73  best=95.83\n",
      "Client  88  cur=93.40  best=93.73\n",
      "Client  89  cur=96.87  best=96.87\n",
      "Client  90  cur=85.60  best= 0.00\n",
      "Client  91  cur=98.90  best=98.90\n",
      "Client  92  cur=98.20  best=98.20\n",
      "Client  93  cur=98.93  best=99.17\n",
      "Client  94  cur=95.77  best=96.10\n",
      "Client  95  cur=93.33  best=93.33\n",
      "Client  96  cur=86.90  best=95.83\n",
      "Client  97  cur=95.93  best=95.97\n",
      "Client  98  cur=94.37  best=94.47\n",
      "Client  99  cur=97.53  best=97.73\n",
      "Round 11 AvgCur=93.24  AvgBest=90.28\n",
      "\n",
      "###### P2-ROUND 12 ######\n",
      "###### P2-ROUND 13 ######\n",
      "###### P2-ROUND 14 ######\n",
      "###### P2-ROUND 15 ######\n",
      "###### P2-ROUND 16 ######\n",
      "###### P2-ROUND 17 ######\n",
      "###### P2-ROUND 18 ######\n",
      "###### P2-ROUND 19 ######\n",
      "###### P2-ROUND 20 ######\n",
      "###### P2-ROUND 21 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0100\n",
      "Init Test Acc:  96.45 Final Test Acc: 95.18\n",
      "Client   0  cur=93.50  best=95.77\n",
      "Client   1  cur=98.90  best=98.90\n",
      "Client   2  cur=95.10  best=98.17\n",
      "Client   3  cur=99.10  best=99.10\n",
      "Client   4  cur=95.83  best=95.83\n",
      "Client   5  cur=95.73  best=96.20\n",
      "Client   6  cur=94.30  best=94.30\n",
      "Client   7  cur=95.53  best=95.97\n",
      "Client   8  cur=95.53  best=95.83\n",
      "Client   9  cur=92.10  best=92.10\n",
      "Client  10  cur=92.77  best=92.90\n",
      "Client  11  cur=97.53  best=98.17\n",
      "Client  12  cur=98.77  best=99.13\n",
      "Client  13  cur=93.20  best=96.20\n",
      "Client  14  cur=93.00  best=93.33\n",
      "Client  15  cur=93.87  best=96.27\n",
      "Client  16  cur=92.63  best=93.00\n",
      "Client  17  cur=95.87  best=95.87\n",
      "Client  18  cur=98.57  best=98.57\n",
      "Client  19  cur=99.13  best=99.13\n",
      "Client  20  cur=92.63  best=92.63\n",
      "Client  21  cur=98.27  best=98.30\n",
      "Client  22  cur=99.13  best=99.17\n",
      "Client  23  cur=94.87  best=94.90\n",
      "Client  24  cur=94.67  best=96.10\n",
      "Client  25  cur=95.77  best=95.87\n",
      "Client  26  cur=99.13  best=99.23\n",
      "Client  27  cur=90.57  best=90.57\n",
      "Client  28  cur=98.20  best=98.20\n",
      "Client  29  cur=96.77  best=97.87\n",
      "Client  30  cur=95.67  best=96.00\n",
      "Client  31  cur=96.37  best=96.53\n",
      "Client  32  cur=97.83  best=98.30\n",
      "Client  33  cur=95.60  best=96.93\n",
      "Client  34  cur=99.00  best=99.17\n",
      "Client  35  cur=98.40  best=98.63\n",
      "Client  36  cur=98.10  best=98.47\n",
      "Client  37  cur=97.07  best=98.23\n",
      "Client  38  cur=96.00  best=96.00\n",
      "Client  39  cur=92.73  best=93.13\n",
      "Client  40  cur=99.10  best=99.17\n",
      "Client  41  cur=92.13  best=93.73\n",
      "Client  42  cur=86.40  best=92.53\n",
      "Client  43  cur=93.00  best=93.43\n",
      "Client  44  cur=98.17  best=98.23\n",
      "Client  45  cur=99.27  best=99.27\n",
      "Client  46  cur=91.37  best=95.97\n",
      "Client  47  cur=86.77  best=96.20\n",
      "Client  48  cur=92.07  best=92.80\n",
      "Client  49  cur=99.03  best=99.17\n",
      "Client  50  cur=95.97  best=96.27\n",
      "Client  51  cur=95.53  best=96.60\n",
      "Client  52  cur=95.50  best=96.60\n",
      "Client  53  cur=78.90  best=93.73\n",
      "Client  54  cur=94.70  best=95.83\n",
      "Client  55  cur=92.77  best=92.77\n",
      "Client  56  cur=98.13  best=98.13\n",
      "Client  57  cur=98.07  best=98.23\n",
      "Client  58  cur=95.53  best=95.70\n",
      "Client  59  cur=92.67  best=93.00\n",
      "Client  60  cur=95.17  best=96.93\n",
      "Client  61  cur=99.33  best=99.40\n",
      "Client  62  cur=98.37  best=98.37\n",
      "Client  63  cur=99.40  best=99.40\n",
      "Client  64  cur=96.17  best=96.60\n",
      "Client  65  cur=95.37  best=96.27\n",
      "Client  66  cur=90.40  best=96.20\n",
      "Client  67  cur=96.97  best=98.17\n",
      "Client  68  cur=95.70  best=96.10\n",
      "Client  69  cur=80.17  best=80.17\n",
      "Client  70  cur=97.80  best=97.80\n",
      "Client  71  cur=91.13  best=93.73\n",
      "Client  72  cur=99.07  best=99.10\n",
      "Client  73  cur=99.20  best=99.20\n",
      "Client  74  cur=96.60  best=96.60\n",
      "Client  75  cur=93.10  best=93.30\n",
      "Client  76  cur=95.97  best=96.27\n",
      "Client  77  cur=98.20  best=98.43\n",
      "Client  78  cur=92.03  best=92.53\n",
      "Client  79  cur=99.20  best=99.20\n",
      "Client  80  cur=91.83  best=93.30\n",
      "Client  81  cur=91.33  best=91.33\n",
      "Client  82  cur=93.57  best=93.57\n",
      "Client  83  cur=93.63  best=95.97\n",
      "Client  84  cur=96.17  best=96.93\n",
      "Client  85  cur=85.43  best=93.30\n",
      "Client  86  cur=96.43  best=96.93\n",
      "Client  87  cur=95.60  best=95.83\n",
      "Client  88  cur=93.37  best=93.73\n",
      "Client  89  cur=96.90  best=96.90\n",
      "Client  90  cur=95.20  best=95.23\n",
      "Client  91  cur=98.97  best=99.03\n",
      "Client  92  cur=98.27  best=98.27\n",
      "Client  93  cur=98.93  best=99.17\n",
      "Client  94  cur=95.77  best=96.10\n",
      "Client  95  cur=94.10  best=95.97\n",
      "Client  96  cur=85.17  best=95.83\n",
      "Client  97  cur=95.93  best=95.97\n",
      "Client  98  cur=94.60  best=94.60\n",
      "Client  99  cur=97.53  best=97.73\n",
      "Round 21 AvgCur=95.11  AvgBest=96.16\n",
      "\n",
      "###### P2-ROUND 22 ######\n",
      "###### P2-ROUND 23 ######\n",
      "###### P2-ROUND 24 ######\n",
      "###### P2-ROUND 25 ######\n",
      "###### P2-ROUND 26 ######\n",
      "###### P2-ROUND 27 ######\n",
      "###### P2-ROUND 28 ######\n",
      "###### P2-ROUND 29 ######\n",
      "###### P2-ROUND 30 ######\n",
      "###### P2-ROUND 31 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0082\n",
      "Init Test Acc:  94.91 Final Test Acc: 94.81\n",
      "Client   0  cur=93.83  best=96.00\n",
      "Client   1  cur=98.77  best=98.90\n",
      "Client   2  cur=94.63  best=98.17\n",
      "Client   3  cur=99.10  best=99.10\n",
      "Client   4  cur=95.77  best=96.03\n",
      "Client   5  cur=95.53  best=96.20\n",
      "Client   6  cur=95.07  best=95.93\n",
      "Client   7  cur=95.90  best=96.00\n",
      "Client   8  cur=95.67  best=95.83\n",
      "Client   9  cur=92.00  best=92.10\n",
      "Client  10  cur=92.87  best=92.90\n",
      "Client  11  cur=97.53  best=98.17\n",
      "Client  12  cur=98.97  best=99.13\n",
      "Client  13  cur=93.23  best=96.20\n",
      "Client  14  cur=92.97  best=93.33\n",
      "Client  15  cur=94.03  best=96.27\n",
      "Client  16  cur=91.37  best=93.00\n",
      "Client  17  cur=96.00  best=96.00\n",
      "Client  18  cur=98.43  best=98.57\n",
      "Client  19  cur=99.13  best=99.17\n",
      "Client  20  cur=92.60  best=92.63\n",
      "Client  21  cur=98.37  best=98.40\n",
      "Client  22  cur=99.13  best=99.17\n",
      "Client  23  cur=94.33  best=94.90\n",
      "Client  24  cur=94.67  best=96.10\n",
      "Client  25  cur=96.03  best=96.03\n",
      "Client  26  cur=99.13  best=99.23\n",
      "Client  27  cur=89.93  best=90.57\n",
      "Client  28  cur=98.50  best=98.50\n",
      "Client  29  cur=96.77  best=97.87\n",
      "Client  30  cur=96.00  best=97.07\n",
      "Client  31  cur=96.73  best=97.07\n",
      "Client  32  cur=98.10  best=98.30\n",
      "Client  33  cur=95.80  best=96.93\n",
      "Client  34  cur=98.93  best=99.17\n",
      "Client  35  cur=98.40  best=98.63\n",
      "Client  36  cur=98.33  best=98.60\n",
      "Client  37  cur=97.27  best=98.23\n",
      "Client  38  cur=96.03  best=96.03\n",
      "Client  39  cur=92.63  best=93.13\n",
      "Client  40  cur=99.10  best=99.17\n",
      "Client  41  cur=92.07  best=93.73\n",
      "Client  42  cur=85.93  best=92.53\n",
      "Client  43  cur=92.90  best=93.43\n",
      "Client  44  cur=98.17  best=98.30\n",
      "Client  45  cur=99.27  best=99.27\n",
      "Client  46  cur=92.63  best=96.00\n",
      "Client  47  cur=88.30  best=96.20\n",
      "Client  48  cur=92.00  best=92.80\n",
      "Client  49  cur=98.93  best=99.17\n",
      "Client  50  cur=96.33  best=97.07\n",
      "Client  51  cur=95.57  best=97.07\n",
      "Client  52  cur=95.93  best=96.60\n",
      "Client  53  cur=77.27  best=93.73\n",
      "Client  54  cur=94.07  best=95.83\n",
      "Client  55  cur=92.53  best=92.77\n",
      "Client  56  cur=98.13  best=98.13\n",
      "Client  57  cur=98.07  best=98.23\n",
      "Client  58  cur=95.53  best=95.70\n",
      "Client  59  cur=92.40  best=93.00\n",
      "Client  60  cur=94.80  best=96.93\n",
      "Client  61  cur=99.27  best=99.40\n",
      "Client  62  cur=98.23  best=98.37\n",
      "Client  63  cur=99.33  best=99.40\n",
      "Client  64  cur=96.03  best=96.60\n",
      "Client  65  cur=95.37  best=96.27\n",
      "Client  66  cur=85.90  best=96.20\n",
      "Client  67  cur=96.90  best=98.17\n",
      "Client  68  cur=96.00  best=96.10\n",
      "Client  69  cur=78.60  best=90.07\n",
      "Client  70  cur=98.07  best=98.10\n",
      "Client  71  cur=91.20  best=93.73\n",
      "Client  72  cur=99.00  best=99.10\n",
      "Client  73  cur=99.17  best=99.20\n",
      "Client  74  cur=96.30  best=96.60\n",
      "Client  75  cur=93.47  best=93.47\n",
      "Client  76  cur=96.07  best=96.27\n",
      "Client  77  cur=98.20  best=98.43\n",
      "Client  78  cur=92.77  best=92.77\n",
      "Client  79  cur=99.13  best=99.20\n",
      "Client  80  cur=91.90  best=93.30\n",
      "Client  81  cur=90.90  best=91.33\n",
      "Client  82  cur=93.57  best=93.57\n",
      "Client  83  cur=93.97  best=95.97\n",
      "Client  84  cur=95.80  best=96.93\n",
      "Client  85  cur=85.87  best=93.30\n",
      "Client  86  cur=96.60  best=96.93\n",
      "Client  87  cur=95.60  best=95.83\n",
      "Client  88  cur=93.70  best=93.73\n",
      "Client  89  cur=96.67  best=96.90\n",
      "Client  90  cur=95.60  best=95.60\n",
      "Client  91  cur=99.20  best=99.20\n",
      "Client  92  cur=98.37  best=98.37\n",
      "Client  93  cur=98.90  best=99.17\n",
      "Client  94  cur=95.83  best=96.10\n",
      "Client  95  cur=94.33  best=96.00\n",
      "Client  96  cur=87.73  best=95.83\n",
      "Client  97  cur=94.43  best=95.97\n",
      "Client  98  cur=94.27  best=94.60\n",
      "Client  99  cur=97.67  best=97.73\n",
      "Round 31 AvgCur=95.08  AvgBest=96.33\n",
      "\n",
      "###### P2-ROUND 32 ######\n",
      "###### P2-ROUND 33 ######\n",
      "###### P2-ROUND 34 ######\n",
      "###### P2-ROUND 35 ######\n",
      "###### P2-ROUND 36 ######\n",
      "###### P2-ROUND 37 ######\n",
      "###### P2-ROUND 38 ######\n",
      "###### P2-ROUND 39 ######\n",
      "###### P2-ROUND 40 ######\n",
      "###### P2-ROUND 41 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0055\n",
      "Init Test Acc:  93.81 Final Test Acc: 95.38\n",
      "Client   0  cur=93.67  best=96.07\n",
      "Client   1  cur=98.97  best=99.37\n",
      "Client   2  cur=94.90  best=98.17\n",
      "Client   3  cur=99.30  best=99.30\n",
      "Client   4  cur=95.77  best=96.10\n",
      "Client   5  cur=95.70  best=96.20\n",
      "Client   6  cur=95.33  best=96.40\n",
      "Client   7  cur=95.73  best=96.00\n",
      "Client   8  cur=95.97  best=96.07\n",
      "Client   9  cur=92.23  best=92.63\n",
      "Client  10  cur=92.97  best=92.97\n",
      "Client  11  cur=97.53  best=98.17\n",
      "Client  12  cur=98.97  best=99.13\n",
      "Client  13  cur=92.93  best=96.20\n",
      "Client  14  cur=92.77  best=93.33\n",
      "Client  15  cur=95.13  best=96.27\n",
      "Client  16  cur=91.37  best=93.00\n",
      "Client  17  cur=95.13  best=96.00\n",
      "Client  18  cur=98.57  best=98.93\n",
      "Client  19  cur=99.13  best=99.17\n",
      "Client  20  cur=92.53  best=92.63\n",
      "Client  21  cur=98.10  best=98.40\n",
      "Client  22  cur=99.13  best=99.17\n",
      "Client  23  cur=94.30  best=94.90\n",
      "Client  24  cur=95.10  best=96.10\n",
      "Client  25  cur=96.10  best=96.10\n",
      "Client  26  cur=99.17  best=99.23\n",
      "Client  27  cur=89.93  best=90.57\n",
      "Client  28  cur=98.50  best=98.50\n",
      "Client  29  cur=96.93  best=97.87\n",
      "Client  30  cur=96.23  best=97.07\n",
      "Client  31  cur=96.73  best=97.07\n",
      "Client  32  cur=98.40  best=98.50\n",
      "Client  33  cur=95.67  best=96.93\n",
      "Client  34  cur=98.87  best=99.37\n",
      "Client  35  cur=98.43  best=98.63\n",
      "Client  36  cur=98.23  best=98.60\n",
      "Client  37  cur=97.43  best=98.23\n",
      "Client  38  cur=95.50  best=96.03\n",
      "Client  39  cur=92.63  best=93.13\n",
      "Client  40  cur=99.13  best=99.37\n",
      "Client  41  cur=92.40  best=93.73\n",
      "Client  42  cur=86.63  best=92.53\n",
      "Client  43  cur=93.03  best=93.43\n",
      "Client  44  cur=98.13  best=98.30\n",
      "Client  45  cur=99.30  best=99.37\n",
      "Client  46  cur=91.93  best=96.00\n",
      "Client  47  cur=88.87  best=96.20\n",
      "Client  48  cur=92.13  best=92.80\n",
      "Client  49  cur=99.07  best=99.17\n",
      "Client  50  cur=96.33  best=97.07\n",
      "Client  51  cur=95.40  best=97.07\n",
      "Client  52  cur=96.20  best=96.60\n",
      "Client  53  cur=76.13  best=93.73\n",
      "Client  54  cur=94.07  best=95.83\n",
      "Client  55  cur=92.97  best=92.97\n",
      "Client  56  cur=98.13  best=98.23\n",
      "Client  57  cur=98.13  best=98.23\n",
      "Client  58  cur=94.70  best=95.70\n",
      "Client  59  cur=93.00  best=93.00\n",
      "Client  60  cur=95.13  best=96.97\n",
      "Client  61  cur=99.33  best=99.40\n",
      "Client  62  cur=98.23  best=98.37\n",
      "Client  63  cur=99.37  best=99.40\n",
      "Client  64  cur=95.77  best=96.97\n",
      "Client  65  cur=93.90  best=96.27\n",
      "Client  66  cur=85.90  best=96.20\n",
      "Client  67  cur=97.13  best=98.17\n",
      "Client  68  cur=95.67  best=96.10\n",
      "Client  69  cur=80.20  best=90.07\n",
      "Client  70  cur=98.03  best=98.10\n",
      "Client  71  cur=91.67  best=93.73\n",
      "Client  72  cur=99.10  best=99.13\n",
      "Client  73  cur=99.20  best=99.20\n",
      "Client  74  cur=96.47  best=96.60\n",
      "Client  75  cur=93.13  best=93.47\n",
      "Client  76  cur=96.43  best=96.97\n",
      "Client  77  cur=98.43  best=98.43\n",
      "Client  78  cur=92.40  best=92.77\n",
      "Client  79  cur=99.20  best=99.23\n",
      "Client  80  cur=91.77  best=93.30\n",
      "Client  81  cur=90.30  best=91.57\n",
      "Client  82  cur=94.20  best=94.20\n",
      "Client  83  cur=93.63  best=95.97\n",
      "Client  84  cur=96.23  best=96.93\n",
      "Client  85  cur=85.00  best=93.30\n",
      "Client  86  cur=96.60  best=96.93\n",
      "Client  87  cur=95.60  best=95.83\n",
      "Client  88  cur=93.70  best=93.73\n",
      "Client  89  cur=96.70  best=96.97\n",
      "Client  90  cur=95.13  best=95.70\n",
      "Client  91  cur=99.20  best=99.30\n",
      "Client  92  cur=98.37  best=98.47\n",
      "Client  93  cur=98.90  best=99.17\n",
      "Client  94  cur=95.63  best=96.10\n",
      "Client  95  cur=94.20  best=96.00\n",
      "Client  96  cur=86.57  best=95.83\n",
      "Client  97  cur=95.47  best=95.97\n",
      "Client  98  cur=94.97  best=94.97\n",
      "Client  99  cur=97.27  best=97.73\n",
      "Round 41 AvgCur=95.10  AvgBest=96.39\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# ---------------------------------------------------------------------\n",
    "#  PRE-REQS (already defined earlier in your code base)\n",
    "#  --------------------------------------------------------------------\n",
    "# clusters               : List[List[int]]   – client IDs per cluster\n",
    "# clients                : List[Client_ClusterFL]\n",
    "# clients[k].net         : CombinedModelMNIST (4-component model)\n",
    "# clients_clust_id       : Dict[client_id → cluster_id]\n",
    "# net_dataidx_map        : Dict[client_id → #samples]  (for weighting)\n",
    "# FedAvg2(state_dict_list, weight_list)      – utility\n",
    "# ---------------------------------------------------------------------\n",
    "\n",
    "import numpy as np, copy, gc\n",
    "\n",
    "# ---------------------------------------------------------------------\n",
    "# 1.  Initialise per-cluster cache (trainable parts only)\n",
    "# ---------------------------------------------------------------------\n",
    "cluster_global_trainable = {\n",
    "    c: {\n",
    "        'encoder':    copy.deepcopy(\n",
    "            clients[clusters[c][0]].net.own_encoder.state_dict()),\n",
    "        'classifier': copy.deepcopy(\n",
    "            clients[clusters[c][0]].net.own_classifier.state_dict()),\n",
    "    }\n",
    "    for c in range(len(clusters))\n",
    "}\n",
    "\n",
    "# ---------------------------------------------------------------------\n",
    "# 2.  Phase-2 FedAvg loop  (encoder + classifier)\n",
    "# ---------------------------------------------------------------------\n",
    "flag = 10                    # print interval after warm-up\n",
    "loss_locals2 = []\n",
    "init_local_tacc2, init_local_tloss2  = [], []\n",
    "final_local_tacc2, final_local_tloss2 = [], []\n",
    "clients2_best_acc2 = [0 for _ in range(args.num_users)]\n",
    "\n",
    "for round_idx in range(41):\n",
    "\n",
    "    print(f'###### P2-ROUND {round_idx+1} ######')\n",
    "\n",
    "    # ----- 2.1  sample participants ----------------------------------\n",
    "    idxs_users = np.random.choice(\n",
    "        range(args.num_users),\n",
    "        int(args.frac * args.num_users),\n",
    "        replace=False\n",
    "    )\n",
    "\n",
    "    # ----- 2.2  broadcast current cluster weights -------------------\n",
    "    for uid in idxs_users:\n",
    "        cid = clients_clust_id[uid]\n",
    "        clients[uid].net.own_encoder.load_state_dict(\n",
    "            cluster_global_trainable[cid]['encoder'])\n",
    "        clients[uid].net.own_classifier.load_state_dict(\n",
    "            cluster_global_trainable[cid]['classifier'])\n",
    "\n",
    "    # ----- 2.3  book-keeping ----------------------------------------\n",
    "    idx_clusters_round = {c: [] for c in range(len(clusters))}\n",
    "    for uid in idxs_users:\n",
    "        idx_clusters_round[clients_clust_id[uid]].append(uid)\n",
    "\n",
    "    # containers for FedAvg\n",
    "    cluster_encoders, cluster_classifiers, cluster_freqs = {}, {}, {}\n",
    "\n",
    "    # ----- 2.4  local training --------------------------------------\n",
    "    for uid in idxs_users:\n",
    "        cid = clients_clust_id[uid]\n",
    "\n",
    "        # metrics before local train\n",
    "        if round_idx % flag == 0:\n",
    "            l0, a0 = clients[uid].eval_test()\n",
    "            init_local_tloss2.append(l0)\n",
    "            init_local_tacc2.append(a0)\n",
    "            clients2_best_acc2[uid] = max(clients2_best_acc2[uid], a0)\n",
    "\n",
    "        # local update (trainable own-encoder + own-classifier)\n",
    "        loss = clients[uid].train2()\n",
    "        loss_locals2.append(loss)\n",
    "\n",
    "        # metrics after local train\n",
    "        l1, a1 = clients[uid].eval_test()\n",
    "        final_local_tloss2.append(l1)\n",
    "        final_local_tacc2.append(a1)\n",
    "        clients2_best_acc2[uid] = max(clients2_best_acc2[uid], a1)\n",
    "\n",
    "        # collect trainable parts for FedAvg\n",
    "        cluster_encoders   .setdefault(cid, []).append(\n",
    "            copy.deepcopy(clients[uid].net.own_encoder.state_dict()))\n",
    "        cluster_classifiers.setdefault(cid, []).append(\n",
    "            copy.deepcopy(clients[uid].net.own_classifier.state_dict()))\n",
    "        cluster_freqs      .setdefault(cid, []).append(\n",
    "            len(net_dataidx_map[uid]))\n",
    "\n",
    "    # ----- 2.5  FedAvg encoder & classifier per cluster --------------\n",
    "    for cid in cluster_encoders:          # same keys as classifiers\n",
    "        total_samples = sum(cluster_freqs[cid])\n",
    "        weights = [n / total_samples for n in cluster_freqs[cid]]\n",
    "\n",
    "        cluster_global_trainable[cid]['encoder'] = \\\n",
    "            FedAvg2(cluster_encoders[cid], weights)\n",
    "\n",
    "        cluster_global_trainable[cid]['classifier'] = \\\n",
    "            FedAvg2(cluster_classifiers[cid], weights)\n",
    "\n",
    "    # ----- 2.6  pretty printing every `flag` rounds ------------------\n",
    "    if round_idx < 4: flag = 1\n",
    "    else:             flag = 10\n",
    "\n",
    "    if round_idx % flag == 0:\n",
    "        print('## END-OF-ROUND-STATS ##')\n",
    "        print(f'Avg train loss: {np.mean(loss_locals2):.4f}')\n",
    "        print(f'Init Test Acc:  {np.mean(init_local_tacc2):.2f}',\n",
    "              f'Final Test Acc: {np.mean(final_local_tacc2):.2f}')\n",
    "\n",
    "        # per-client summary\n",
    "        cur_acc = []\n",
    "        for uid in range(args.num_users):\n",
    "            _, acc = clients[uid].eval_test()\n",
    "            cur_acc.append(acc)\n",
    "            print(f'Client {uid:3d}  cur={acc:5.2f}  best={clients2_best_acc2[uid]:5.2f}')\n",
    "\n",
    "        print(f'Round {round_idx+1} AvgCur={np.mean(cur_acc):.2f}  '\n",
    "              f'AvgBest={np.mean(clients2_best_acc2):.2f}\\n')\n",
    "\n",
    "        # clear trackers for next flagged round\n",
    "        loss_locals2.clear()\n",
    "        init_local_tacc2.clear(); init_local_tloss2.clear()\n",
    "        final_local_tacc2.clear(); final_local_tloss2.clear()\n",
    "        gc.collect()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "###### P2-ROUND 1 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0104\n",
      "Init Test Acc:  93.79 Final Test Acc: 92.73\n",
      "Client   0  cur=98.63  best=98.63\n",
      "Client   1  cur=98.83  best=99.03\n",
      "Client   2  cur=99.27  best=99.27\n",
      "Client   3  cur=92.87  best=96.67\n",
      "Client   4  cur=84.90  best=90.63\n",
      "Client   5  cur=98.93  best=99.17\n",
      "Client   6  cur=98.70  best=99.07\n",
      "Client   7  cur=90.90  best=93.43\n",
      "Client   8  cur=90.80  best=90.80\n",
      "Client   9  cur=99.03  best=99.03\n",
      "Client  10  cur=97.53  best=98.90\n",
      "Client  11  cur=99.00  best=99.00\n",
      "Client  12  cur=96.23  best=96.70\n",
      "Client  13  cur=98.67  best=98.87\n",
      "Client  14  cur=99.07  best=99.07\n",
      "Client  15  cur=95.77  best=96.67\n",
      "Client  16  cur=89.80  best=90.37\n",
      "Client  17  cur=97.17  best=99.37\n",
      "Client  18  cur=96.50  best=99.37\n",
      "Client  19  cur=98.97  best=98.97\n",
      "Client  20  cur=87.20  best=90.63\n",
      "Client  21  cur=96.27  best=96.27\n",
      "Client  22  cur=91.97  best=92.63\n",
      "Client  23  cur=96.97  best=96.97\n",
      "Client  24  cur=88.83  best=90.87\n",
      "Client  25  cur=86.30  best=90.73\n",
      "Client  26  cur=92.23  best=92.87\n",
      "Client  27  cur=96.23  best=99.37\n",
      "Client  28  cur=90.07  best=92.87\n",
      "Client  29  cur=96.57  best=96.93\n",
      "Client  30  cur=88.03  best=90.87\n",
      "Client  31  cur=96.10  best=96.67\n",
      "Client  32  cur=99.23  best=99.37\n",
      "Client  33  cur=90.97  best=90.97\n",
      "Client  34  cur=96.20  best=96.87\n",
      "Client  35  cur=96.37  best=96.73\n",
      "Client  36  cur=95.57  best=96.47\n",
      "Client  37  cur=95.20  best=96.87\n",
      "Client  38  cur=98.97  best=99.00\n",
      "Client  39  cur=88.60  best=90.03\n",
      "Client  40  cur=98.67  best=99.07\n",
      "Client  41  cur=91.17  best=93.00\n",
      "Client  42  cur=90.40  best=90.87\n",
      "Client  43  cur=89.97  best=92.87\n",
      "Client  44  cur=88.37  best=92.40\n",
      "Client  45  cur=98.87  best=99.07\n",
      "Client  46  cur=90.53  best=90.73\n",
      "Client  47  cur=99.20  best=99.37\n",
      "Client  48  cur=98.90  best=98.97\n",
      "Client  49  cur=87.83  best=88.67\n",
      "Client  50  cur=97.83  best=99.30\n",
      "Client  51  cur=89.37  best=93.00\n",
      "Client  52  cur=94.67  best=96.77\n",
      "Client  53  cur=85.67  best=90.87\n",
      "Client  54  cur=91.97  best=93.43\n",
      "Client  55  cur=89.80  best=90.87\n",
      "Client  56  cur=94.63  best=96.77\n",
      "Client  57  cur=86.87  best=92.87\n",
      "Client  58  cur=99.17  best=99.37\n",
      "Client  59  cur=89.53  best=90.87\n",
      "Client  60  cur=99.10  best=99.10\n",
      "Client  61  cur=96.63  best=96.93\n",
      "Client  62  cur=95.90  best=96.67\n",
      "Client  63  cur=99.27  best=99.27\n",
      "Client  64  cur=99.27  best=99.30\n",
      "Client  65  cur=92.07  best=92.63\n",
      "Client  66  cur=84.23  best=90.77\n",
      "Client  67  cur=98.83  best=99.00\n",
      "Client  68  cur=85.23  best=90.87\n",
      "Client  69  cur=95.83  best=96.93\n",
      "Client  70  cur=98.80  best=98.97\n",
      "Client  71  cur=92.40  best=92.87\n",
      "Client  72  cur=99.27  best=99.37\n",
      "Client  73  cur=90.33  best=90.87\n",
      "Client  74  cur=88.47  best=92.87\n",
      "Client  75  cur=96.40  best=96.67\n",
      "Client  76  cur=91.80  best=93.00\n",
      "Client  77  cur=83.63  best=90.73\n",
      "Client  78  cur=96.80  best=96.97\n",
      "Client  79  cur=96.57  best=96.87\n",
      "Client  80  cur=98.87  best=99.07\n",
      "Client  81  cur=96.87  best=96.87\n",
      "Client  82  cur=90.33  best=91.40\n",
      "Client  83  cur=98.37  best=99.23\n",
      "Client  84  cur=89.53  best=90.63\n",
      "Client  85  cur=89.73  best=90.03\n",
      "Client  86  cur=98.97  best=99.00\n",
      "Client  87  cur=98.90  best=99.37\n",
      "Client  88  cur=99.03  best=99.03\n",
      "Client  89  cur=98.87  best=99.30\n",
      "Client  90  cur=92.60  best=93.43\n",
      "Client  91  cur=95.50  best=96.93\n",
      "Client  92  cur=99.20  best=99.30\n",
      "Client  93  cur=90.30  best=91.43\n",
      "Client  94  cur=94.37  best=94.53\n",
      "Client  95  cur=96.93  best=98.00\n",
      "Client  96  cur=95.63  best=96.70\n",
      "Client  97  cur=99.30  best=99.33\n",
      "Client  98  cur=99.17  best=99.30\n",
      "Client  99  cur=99.23  best=99.33\n",
      "Round 1 AvgCur=94.45  AvgBest=95.69\n",
      "\n",
      "###### P2-ROUND 2 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0079\n",
      "Init Test Acc:  95.98 Final Test Acc: 94.93\n",
      "Client   0  cur=98.63  best=98.63\n",
      "Client   1  cur=98.83  best=99.03\n",
      "Client   2  cur=99.27  best=99.27\n",
      "Client   3  cur=91.83  best=96.80\n",
      "Client   4  cur=84.90  best=90.63\n",
      "Client   5  cur=98.93  best=99.17\n",
      "Client   6  cur=98.70  best=99.07\n",
      "Client   7  cur=90.90  best=93.43\n",
      "Client   8  cur=90.80  best=90.80\n",
      "Client   9  cur=99.03  best=99.03\n",
      "Client  10  cur=97.53  best=98.90\n",
      "Client  11  cur=99.00  best=99.00\n",
      "Client  12  cur=96.23  best=96.70\n",
      "Client  13  cur=98.67  best=98.87\n",
      "Client  14  cur=99.07  best=99.07\n",
      "Client  15  cur=95.77  best=96.67\n",
      "Client  16  cur=89.80  best=90.37\n",
      "Client  17  cur=97.17  best=99.37\n",
      "Client  18  cur=96.50  best=99.37\n",
      "Client  19  cur=98.97  best=98.97\n",
      "Client  20  cur=87.20  best=90.63\n",
      "Client  21  cur=96.27  best=96.27\n",
      "Client  22  cur=91.97  best=92.63\n",
      "Client  23  cur=96.97  best=96.97\n",
      "Client  24  cur=88.83  best=90.87\n",
      "Client  25  cur=86.30  best=90.73\n",
      "Client  26  cur=92.23  best=92.87\n",
      "Client  27  cur=96.23  best=99.37\n",
      "Client  28  cur=87.90  best=92.87\n",
      "Client  29  cur=96.57  best=96.93\n",
      "Client  30  cur=88.10  best=90.87\n",
      "Client  31  cur=96.10  best=96.67\n",
      "Client  32  cur=99.23  best=99.37\n",
      "Client  33  cur=90.97  best=90.97\n",
      "Client  34  cur=96.33  best=96.87\n",
      "Client  35  cur=95.90  best=96.80\n",
      "Client  36  cur=95.70  best=96.80\n",
      "Client  37  cur=95.17  best=96.87\n",
      "Client  38  cur=98.97  best=99.00\n",
      "Client  39  cur=88.60  best=90.03\n",
      "Client  40  cur=98.67  best=99.07\n",
      "Client  41  cur=91.17  best=93.00\n",
      "Client  42  cur=90.40  best=90.87\n",
      "Client  43  cur=89.97  best=92.87\n",
      "Client  44  cur=88.37  best=92.40\n",
      "Client  45  cur=98.87  best=99.07\n",
      "Client  46  cur=90.53  best=90.73\n",
      "Client  47  cur=99.20  best=99.37\n",
      "Client  48  cur=98.90  best=98.97\n",
      "Client  49  cur=87.83  best=88.67\n",
      "Client  50  cur=97.83  best=99.30\n",
      "Client  51  cur=89.37  best=93.00\n",
      "Client  52  cur=94.67  best=96.77\n",
      "Client  53  cur=85.67  best=90.87\n",
      "Client  54  cur=91.97  best=93.43\n",
      "Client  55  cur=89.80  best=90.87\n",
      "Client  56  cur=94.63  best=96.77\n",
      "Client  57  cur=86.87  best=92.87\n",
      "Client  58  cur=99.20  best=99.37\n",
      "Client  59  cur=89.83  best=90.87\n",
      "Client  60  cur=99.10  best=99.10\n",
      "Client  61  cur=96.63  best=96.93\n",
      "Client  62  cur=95.90  best=96.67\n",
      "Client  63  cur=99.10  best=99.27\n",
      "Client  64  cur=99.27  best=99.30\n",
      "Client  65  cur=93.07  best=93.07\n",
      "Client  66  cur=84.23  best=90.77\n",
      "Client  67  cur=98.83  best=99.00\n",
      "Client  68  cur=85.23  best=90.87\n",
      "Client  69  cur=95.83  best=96.93\n",
      "Client  70  cur=98.80  best=98.97\n",
      "Client  71  cur=92.40  best=92.87\n",
      "Client  72  cur=99.27  best=99.37\n",
      "Client  73  cur=90.33  best=90.87\n",
      "Client  74  cur=88.47  best=92.87\n",
      "Client  75  cur=96.40  best=96.67\n",
      "Client  76  cur=92.00  best=93.00\n",
      "Client  77  cur=83.63  best=90.73\n",
      "Client  78  cur=97.03  best=97.03\n",
      "Client  79  cur=96.57  best=96.87\n",
      "Client  80  cur=98.87  best=99.07\n",
      "Client  81  cur=96.77  best=96.87\n",
      "Client  82  cur=90.33  best=91.40\n",
      "Client  83  cur=98.37  best=99.23\n",
      "Client  84  cur=89.53  best=90.63\n",
      "Client  85  cur=89.73  best=90.03\n",
      "Client  86  cur=98.97  best=99.00\n",
      "Client  87  cur=98.90  best=99.37\n",
      "Client  88  cur=99.03  best=99.03\n",
      "Client  89  cur=98.87  best=99.30\n",
      "Client  90  cur=92.60  best=93.43\n",
      "Client  91  cur=95.50  best=96.93\n",
      "Client  92  cur=99.17  best=99.30\n",
      "Client  93  cur=90.30  best=91.43\n",
      "Client  94  cur=94.37  best=94.53\n",
      "Client  95  cur=97.27  best=98.90\n",
      "Client  96  cur=96.03  best=96.80\n",
      "Client  97  cur=99.30  best=99.33\n",
      "Client  98  cur=99.17  best=99.30\n",
      "Client  99  cur=99.23  best=99.33\n",
      "Round 2 AvgCur=94.44  AvgBest=95.71\n",
      "\n",
      "###### P2-ROUND 3 ######\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[28], line 40\u001b[0m\n\u001b[1;32m     37\u001b[0m     clients2_best_acc2[uid] \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mmax\u001b[39m(clients2_best_acc2[uid], a0)\n\u001b[1;32m     39\u001b[0m \u001b[38;5;66;03m# local update (trainable own-encoder + own-classifier)\u001b[39;00m\n\u001b[0;32m---> 40\u001b[0m loss \u001b[38;5;241m=\u001b[39m \u001b[43mclients\u001b[49m\u001b[43m[\u001b[49m\u001b[43muid\u001b[49m\u001b[43m]\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mtrain2\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     41\u001b[0m loss_locals2\u001b[38;5;241m.\u001b[39mappend(loss)\n\u001b[1;32m     43\u001b[0m \u001b[38;5;66;03m# metrics after local train\u001b[39;00m\n",
      "File \u001b[0;32m~/Misc/PACFLComboNB/Flag/src/client/client_cluster_fl.py:91\u001b[0m, in \u001b[0;36mClient_ClusterFL.train2\u001b[0;34m(self, is_print)\u001b[0m\n\u001b[1;32m     89\u001b[0m optimizer\u001b[38;5;241m.\u001b[39mzero_grad()\n\u001b[1;32m     90\u001b[0m logits \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mnet(images)\n\u001b[0;32m---> 91\u001b[0m loss   \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mloss_func\u001b[49m\u001b[43m(\u001b[49m\u001b[43mlogits\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mlabels\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     92\u001b[0m loss\u001b[38;5;241m.\u001b[39mbackward()\n\u001b[1;32m     93\u001b[0m optimizer\u001b[38;5;241m.\u001b[39mstep()\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/module.py:1532\u001b[0m, in \u001b[0;36mModule._wrapped_call_impl\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1530\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_compiled_call_impl(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)  \u001b[38;5;66;03m# type: ignore[misc]\u001b[39;00m\n\u001b[1;32m   1531\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[0;32m-> 1532\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_call_impl\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/module.py:1541\u001b[0m, in \u001b[0;36mModule._call_impl\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1536\u001b[0m \u001b[38;5;66;03m# If we don't have any hooks, we want to skip the rest of the logic in\u001b[39;00m\n\u001b[1;32m   1537\u001b[0m \u001b[38;5;66;03m# this function, and just call forward.\u001b[39;00m\n\u001b[1;32m   1538\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m (\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_pre_hooks\n\u001b[1;32m   1539\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_backward_hooks\n\u001b[1;32m   1540\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_forward_pre_hooks):\n\u001b[0;32m-> 1541\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mforward_call\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1543\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m   1544\u001b[0m     result \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/loss.py:1185\u001b[0m, in \u001b[0;36mCrossEntropyLoss.forward\u001b[0;34m(self, input, target)\u001b[0m\n\u001b[1;32m   1184\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mforward\u001b[39m(\u001b[38;5;28mself\u001b[39m, \u001b[38;5;28minput\u001b[39m: Tensor, target: Tensor) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m Tensor:\n\u001b[0;32m-> 1185\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mF\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mcross_entropy\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43minput\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtarget\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mweight\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mweight\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m   1186\u001b[0m \u001b[43m                           \u001b[49m\u001b[43mignore_index\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mignore_index\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mreduction\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mreduction\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m   1187\u001b[0m \u001b[43m                           \u001b[49m\u001b[43mlabel_smoothing\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mlabel_smoothing\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/functional.py:3086\u001b[0m, in \u001b[0;36mcross_entropy\u001b[0;34m(input, target, weight, size_average, ignore_index, reduce, reduction, label_smoothing)\u001b[0m\n\u001b[1;32m   3084\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m size_average \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m \u001b[38;5;129;01mor\u001b[39;00m reduce \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[1;32m   3085\u001b[0m     reduction \u001b[38;5;241m=\u001b[39m _Reduction\u001b[38;5;241m.\u001b[39mlegacy_get_string(size_average, reduce)\n\u001b[0;32m-> 3086\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mtorch\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_C\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_nn\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mcross_entropy_loss\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43minput\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtarget\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mweight\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m_Reduction\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mget_enum\u001b[49m\u001b[43m(\u001b[49m\u001b[43mreduction\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mignore_index\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mlabel_smoothing\u001b[49m\u001b[43m)\u001b[49m\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "h\n"
     ]
    }
   ],
   "source": [
    "print(\"h\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
