{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The autoreload extension is already loaded. To reload it, use:\n",
      "  %reload_ext autoreload\n",
      "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='svhn', 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='svhn'\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": 3,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "partition: flag-non-iid\n",
      "Data statistics Train:\n",
      " {0: {2: 7, 3: 532, 4: 167}, 1: {2: 85, 3: 12, 9: 163}, 2: {0: 755, 6: 397, 7: 386}, 3: {1: 68, 5: 1286, 8: 124}, 4: {1: 126, 5: 206, 8: 71}, 5: {1: 13, 4: 51, 8: 370}, 6: {2: 173, 3: 99, 4: 40}, 7: {2: 361, 3: 271, 4: 261}, 8: {1: 64, 4: 217, 8: 58}, 9: {2: 8, 3: 95, 9: 109}, 10: {0: 182, 6: 193, 7: 610}, 11: {2: 188, 3: 200, 4: 1008}, 12: {0: 187, 6: 359, 7: 33}, 13: {0: 339, 6: 113, 7: 109}, 14: {2: 201, 3: 595, 4: 27}, 15: {2: 130, 3: 28, 9: 50}, 16: {1: 679, 5: 154, 8: 195}, 17: {1: 88, 5: 60, 8: 118}, 18: {1: 477, 5: 137, 8: 60}, 19: {2: 194, 3: 49, 4: 94}, 20: {2: 12, 3: 35, 9: 488}, 21: {1: 304, 4: 192, 8: 62}, 22: {1: 355, 5: 443, 8: 520}, 23: {2: 641, 3: 375, 4: 187}, 24: {2: 40, 3: 472, 9: 70}, 25: {1: 426, 4: 40, 8: 14}, 26: {1: 1581, 5: 28, 8: 10}, 27: {2: 255, 3: 6, 4: 113}, 28: {2: 1047, 3: 507, 9: 237}, 29: {0: 72, 6: 319, 7: 957}, 30: {2: 110, 3: 286, 9: 209}, 31: {1: 64, 5: 219, 8: 333}, 32: {1: 347, 4: 465, 8: 458}, 33: {1: 31, 4: 58, 8: 166}, 34: {0: 423, 6: 199, 7: 756}, 35: {1: 354, 5: 70, 8: 21}, 36: {1: 361, 4: 20, 8: 47}, 37: {0: 7, 6: 101, 7: 199}, 38: {2: 219, 3: 581, 9: 66}, 39: {2: 90, 3: 613, 9: 24}, 40: {1: 26, 4: 82, 8: 49}, 41: {2: 1055, 3: 135, 4: 281}, 42: {2: 208, 3: 6, 9: 1091}, 43: {1: 390, 5: 202, 8: 99}, 44: {1: 1062, 4: 161, 8: 24}, 45: {1: 296, 5: 2, 8: 37}, 46: {0: 378, 6: 256, 7: 121}, 47: {2: 154, 3: 399, 4: 305}, 48: {2: 166, 3: 108, 9: 132}, 49: {0: 22, 6: 216, 7: 12}, 50: {2: 235, 3: 115, 4: 458}, 51: {1: 302, 5: 539, 8: 64}, 52: {1: 191, 5: 354, 8: 171}, 53: {0: 458, 6: 175, 7: 789}, 54: {2: 36, 3: 57, 9: 50}, 55: {1: 191, 4: 243, 8: 56}, 56: {1: 191, 5: 781, 8: 212}, 57: {0: 238, 6: 656, 7: 57}, 58: {1: 566, 4: 128, 8: 159}, 59: {2: 141, 3: 323, 4: 174}, 60: {1: 560, 4: 9, 8: 30}, 61: {2: 540, 3: 350, 9: 21}, 62: {2: 247, 3: 351, 9: 519}, 63: {1: 149, 4: 8, 8: 19}, 64: {0: 32, 6: 170, 7: 136}, 65: {2: 309, 3: 125, 4: 26}, 66: {0: 644, 6: 148, 7: 126}, 67: {2: 246, 3: 395, 9: 41}, 68: {1: 485, 5: 121, 8: 34}, 69: {1: 57, 5: 818, 8: 80}, 70: {1: 13, 5: 430, 8: 35}, 71: {1: 779, 5: 372, 8: 116}, 72: {2: 299, 3: 104, 4: 519}, 73: {1: 853, 4: 177, 8: 117}, 74: {0: 88, 6: 929, 7: 104}, 75: {1: 62, 4: 145, 8: 42}, 76: {2: 26, 3: 226, 4: 338}, 77: {2: 263, 3: 507, 9: 151}, 78: {2: 86, 3: 47, 4: 123}, 79: {0: 26, 6: 408, 7: 10}, 80: {0: 217, 6: 127, 7: 17}, 81: {0: 502, 6: 47, 7: 267}, 82: {0: 89, 6: 95, 7: 101}, 83: {1: 102, 4: 61, 8: 1}, 84: {0: 92, 6: 248, 7: 181}, 85: {1: 290, 4: 208, 8: 185}, 86: {2: 583, 3: 58, 9: 807}, 87: {2: 199, 3: 91, 4: 2}, 88: {1: 92, 4: 7, 8: 83}, 89: {1: 16, 5: 485, 8: 87}, 90: {1: 713, 5: 175, 8: 4}, 91: {2: 212, 3: 14, 4: 153}, 92: {2: 172, 3: 182, 4: 744}, 93: {2: 241, 3: 87, 9: 413}, 94: {2: 546, 3: 33, 9: 18}, 95: {1: 247, 4: 142, 8: 43}, 96: {0: 194, 6: 415, 7: 541}, 97: {0: 3, 6: 156, 7: 83}, 98: {1: 890, 4: 17, 8: 671}, 99: {2: 860, 3: 28, 4: 7}} \n",
      "\n",
      "Data statistics Test:\n",
      " {0: {2: 4149, 3: 2882, 4: 2523}, 1: {2: 4149, 3: 2882, 9: 1595}, 2: {0: 1744, 6: 1977, 7: 2019}, 3: {1: 5099, 5: 2384, 8: 1660}, 4: {1: 5099, 5: 2384, 8: 1660}, 5: {1: 5099, 4: 2523, 8: 1660}, 6: {2: 4149, 3: 2882, 4: 2523}, 7: {2: 4149, 3: 2882, 4: 2523}, 8: {1: 5099, 4: 2523, 8: 1660}, 9: {2: 4149, 3: 2882, 9: 1595}, 10: {0: 1744, 6: 1977, 7: 2019}, 11: {2: 4149, 3: 2882, 4: 2523}, 12: {0: 1744, 6: 1977, 7: 2019}, 13: {0: 1744, 6: 1977, 7: 2019}, 14: {2: 4149, 3: 2882, 4: 2523}, 15: {2: 4149, 3: 2882, 9: 1595}, 16: {1: 5099, 5: 2384, 8: 1660}, 17: {1: 5099, 5: 2384, 8: 1660}, 18: {1: 5099, 5: 2384, 8: 1660}, 19: {2: 4149, 3: 2882, 4: 2523}, 20: {2: 4149, 3: 2882, 9: 1595}, 21: {1: 5099, 4: 2523, 8: 1660}, 22: {1: 5099, 5: 2384, 8: 1660}, 23: {2: 4149, 3: 2882, 4: 2523}, 24: {2: 4149, 3: 2882, 9: 1595}, 25: {1: 5099, 4: 2523, 8: 1660}, 26: {1: 5099, 5: 2384, 8: 1660}, 27: {2: 4149, 3: 2882, 4: 2523}, 28: {2: 4149, 3: 2882, 9: 1595}, 29: {0: 1744, 6: 1977, 7: 2019}, 30: {2: 4149, 3: 2882, 9: 1595}, 31: {1: 5099, 5: 2384, 8: 1660}, 32: {1: 5099, 4: 2523, 8: 1660}, 33: {1: 5099, 4: 2523, 8: 1660}, 34: {0: 1744, 6: 1977, 7: 2019}, 35: {1: 5099, 5: 2384, 8: 1660}, 36: {1: 5099, 4: 2523, 8: 1660}, 37: {0: 1744, 6: 1977, 7: 2019}, 38: {2: 4149, 3: 2882, 9: 1595}, 39: {2: 4149, 3: 2882, 9: 1595}, 40: {1: 5099, 4: 2523, 8: 1660}, 41: {2: 4149, 3: 2882, 4: 2523}, 42: {2: 4149, 3: 2882, 9: 1595}, 43: {1: 5099, 5: 2384, 8: 1660}, 44: {1: 5099, 4: 2523, 8: 1660}, 45: {1: 5099, 5: 2384, 8: 1660}, 46: {0: 1744, 6: 1977, 7: 2019}, 47: {2: 4149, 3: 2882, 4: 2523}, 48: {2: 4149, 3: 2882, 9: 1595}, 49: {0: 1744, 6: 1977, 7: 2019}, 50: {2: 4149, 3: 2882, 4: 2523}, 51: {1: 5099, 5: 2384, 8: 1660}, 52: {1: 5099, 5: 2384, 8: 1660}, 53: {0: 1744, 6: 1977, 7: 2019}, 54: {2: 4149, 3: 2882, 9: 1595}, 55: {1: 5099, 4: 2523, 8: 1660}, 56: {1: 5099, 5: 2384, 8: 1660}, 57: {0: 1744, 6: 1977, 7: 2019}, 58: {1: 5099, 4: 2523, 8: 1660}, 59: {2: 4149, 3: 2882, 4: 2523}, 60: {1: 5099, 4: 2523, 8: 1660}, 61: {2: 4149, 3: 2882, 9: 1595}, 62: {2: 4149, 3: 2882, 9: 1595}, 63: {1: 5099, 4: 2523, 8: 1660}, 64: {0: 1744, 6: 1977, 7: 2019}, 65: {2: 4149, 3: 2882, 4: 2523}, 66: {0: 1744, 6: 1977, 7: 2019}, 67: {2: 4149, 3: 2882, 9: 1595}, 68: {1: 5099, 5: 2384, 8: 1660}, 69: {1: 5099, 5: 2384, 8: 1660}, 70: {1: 5099, 5: 2384, 8: 1660}, 71: {1: 5099, 5: 2384, 8: 1660}, 72: {2: 4149, 3: 2882, 4: 2523}, 73: {1: 5099, 4: 2523, 8: 1660}, 74: {0: 1744, 6: 1977, 7: 2019}, 75: {1: 5099, 4: 2523, 8: 1660}, 76: {2: 4149, 3: 2882, 4: 2523}, 77: {2: 4149, 3: 2882, 9: 1595}, 78: {2: 4149, 3: 2882, 4: 2523}, 79: {0: 1744, 6: 1977, 7: 2019}, 80: {0: 1744, 6: 1977, 7: 2019}, 81: {0: 1744, 6: 1977, 7: 2019}, 82: {0: 1744, 6: 1977, 7: 2019}, 83: {1: 5099, 4: 2523, 8: 1660}, 84: {0: 1744, 6: 1977, 7: 2019}, 85: {1: 5099, 4: 2523, 8: 1660}, 86: {2: 4149, 3: 2882, 9: 1595}, 87: {2: 4149, 3: 2882, 4: 2523}, 88: {1: 5099, 4: 2523, 8: 1660}, 89: {1: 5099, 5: 2384, 8: 1660}, 90: {1: 5099, 5: 2384, 8: 1660}, 91: {2: 4149, 3: 2882, 4: 2523}, 92: {2: 4149, 3: 2882, 4: 2523}, 93: {2: 4149, 3: 2882, 9: 1595}, 94: {2: 4149, 3: 2882, 9: 1595}, 95: {1: 5099, 4: 2523, 8: 1660}, 96: {0: 1744, 6: 1977, 7: 2019}, 97: {0: 1744, 6: 1977, 7: 2019}, 98: {1: 5099, 4: 2523, 8: 1660}, 99: {2: 4149, 3: 2882, 4: 2523}} \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": 4,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "len train_ds_global: 73257\n",
      "len test_ds_global: 26032\n",
      "MODEL: simple-cnn, Dataset: svhn\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): Sequential(\n",
      "    (0): Dropout(p=0.5, inplace=False)\n",
      "    (1): Linear(in_features=256, out_features=10, bias=True)\n",
      "  )\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.1.weight torch.Size([10, 256])\n",
      "classifier.1.bias torch.Size([10])\n",
      "537610\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [  7 532 167]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [ 85  12 163]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [755 397 386]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [  68 1286  124]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [126 206  71]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [ 13  51 370]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [173  99  40]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [361 271 261]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [ 64 217  58]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [  8  95 109]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [182 193 610]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [ 188  200 1008]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [187 359  33]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [339 113 109]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [201 595  27]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [130  28  50]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [679 154 195]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [ 88  60 118]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [477 137  60]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [194  49  94]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [ 12  35 488]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [304 192  62]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [355 443 520]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [641 375 187]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [ 40 472  70]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [426  40  14]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [1581   28   10]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [255   6 113]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [1047  507  237]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [ 72 319 957]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [110 286 209]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [ 64 219 333]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [347 465 458]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [ 31  58 166]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [423 199 756]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [354  70  21]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [361  20  47]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [  7 101 199]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [219 581  66]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [ 90 613  24]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [26 82 49]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [1055  135  281]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [ 208    6 1091]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [390 202  99]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [1062  161   24]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [296   2  37]\n",
      "Shape of U: (3072, 8)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [378 256 121]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [154 399 305]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [166 108 132]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [ 22 216  12]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [235 115 458]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [302 539  64]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [191 354 171]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [458 175 789]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [36 57 50]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [191 243  56]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [191 781 212]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [238 656  57]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [566 128 159]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [141 323 174]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [560   9  30]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [540 350  21]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [247 351 519]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [149   8  19]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [ 32 170 136]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [309 125  26]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [644 148 126]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [246 395  41]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [485 121  34]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [ 57 818  80]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [ 13 430  35]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [779 372 116]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [299 104 519]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [853 177 117]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [ 88 929 104]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [ 62 145  42]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [ 26 226 338]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [263 507 151]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [ 86  47 123]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [ 26 408  10]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [217 127  17]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [502  47 267]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [ 89  95 101]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [102  61   1]\n",
      "Shape of U: (3072, 7)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [ 92 248 181]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [290 208 185]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [583  58 807]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [199  91   2]\n",
      "Shape of U: (3072, 8)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [92  7 83]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [ 16 485  87]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 5 8], Counts: [713 175   4]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [212  14 153]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [172 182 744]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [241  87 413]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 9], Counts: [546  33  18]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [247 142  43]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [194 415 541]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [0 6 7], Counts: [  3 156  83]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [1 4 8], Counts: [890  17 671]\n",
      "Shape of U: (3072, 9)\n",
      "Using downloaded and verified file: ../data/train_32x32.mat\n",
      "Using downloaded and verified file: ../data/test_32x32.mat\n",
      "Labels: [2 3 4], Counts: [860  28   7]\n",
      "Shape of U: (3072, 9)\n",
      "###### Gradient data ROUND 1 ######\n",
      "Grad_similarities:\n",
      "[[  0.0000  83.2983  97.5408 ...  95.7983  94.1308  79.5780]\n",
      " [ 83.2983   0.0000  99.9146 ...  91.8449  94.8223  69.5325]\n",
      " [ 97.5408  99.9146   0.0000 ...  66.5642  94.7385  99.0825]\n",
      " ...\n",
      " [ 95.7983  91.8449  66.5642 ...   0.0000 100.4862  91.9827]\n",
      " [ 94.1308  94.8223  94.7385 ... 100.4862   0.0000  92.5409]\n",
      " [ 79.5780  69.5325  99.0825 ...  91.9827  92.5409   0.0000]]\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": 5,
   "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 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 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 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 1.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 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 1.0000\n",
      " 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.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": 6,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Round 1.5\n",
      "Breaking HC\n",
      "\n",
      "Adjacency Matrix\n",
      "[[0.0000 0.3420 1.0000 ... 1.0000 0.6705 0.0094]\n",
      " [0.3420 0.0000 1.0000 ... 1.0000 1.0000 0.3390]\n",
      " [1.0000 1.0000 0.0000 ... 0.0077 1.0000 1.0000]\n",
      " ...\n",
      " [1.0000 1.0000 0.0077 ... 0.0000 1.0000 1.0000]\n",
      " [0.6705 1.0000 1.0000 ... 1.0000 0.0000 0.6735]\n",
      " [0.0094 0.3390 1.0000 ... 1.0000 0.6735 0.0000]]\n",
      "\n",
      "Clusters: \n",
      "[[0, 6, 7, 23, 41, 47, 50, 92, 11, 59, 72, 19, 76, 14, 65, 78, 27, 91, 99, 87, 1, 42, 20, 24, 28, 62, 77, 86, 93, 38, 67, 30, 61, 48, 39, 54, 94, 15, 9], [2, 34, 53, 96, 46, 29, 10, 74, 57, 13, 84, 82, 66, 81, 12, 80, 64, 49, 97, 37, 79], [3, 18, 22, 4, 17, 16, 43, 52, 51, 56, 71, 31, 69, 68, 35, 26, 90, 70, 89, 45, 5, 32, 58, 44, 40, 33, 8, 21, 55, 73, 85, 95, 75, 98, 36, 25, 88, 60, 63, 83]]\n",
      "\n",
      "Number of Clusters 3\n",
      "\n",
      "Cluster 0: 39 \n",
      "Cluster 1: 21 \n",
      "Cluster 2: 40 \n",
      "Clients: Cluster_ID \n",
      "{0: 0, 1: 0, 2: 1, 3: 2, 4: 2, 5: 2, 6: 0, 7: 0, 8: 2, 9: 0, 10: 1, 11: 0, 12: 1, 13: 1, 14: 0, 15: 0, 16: 2, 17: 2, 18: 2, 19: 0, 20: 0, 21: 2, 22: 2, 23: 0, 24: 0, 25: 2, 26: 2, 27: 0, 28: 0, 29: 1, 30: 0, 31: 2, 32: 2, 33: 2, 34: 1, 35: 2, 36: 2, 37: 1, 38: 0, 39: 0, 40: 2, 41: 0, 42: 0, 43: 2, 44: 2, 45: 2, 46: 1, 47: 0, 48: 0, 49: 1, 50: 0, 51: 2, 52: 2, 53: 1, 54: 0, 55: 2, 56: 2, 57: 1, 58: 2, 59: 0, 60: 2, 61: 0, 62: 0, 63: 2, 64: 1, 65: 0, 66: 1, 67: 0, 68: 2, 69: 2, 70: 2, 71: 2, 72: 0, 73: 2, 74: 1, 75: 2, 76: 0, 77: 0, 78: 0, 79: 1, 80: 1, 81: 1, 82: 1, 83: 2, 84: 1, 85: 2, 86: 0, 87: 0, 88: 2, 89: 2, 90: 2, 91: 0, 92: 0, 93: 0, 94: 0, 95: 2, 96: 1, 97: 1, 98: 2, 99: 0}\n",
      "Cluster 0, First Client 0: {2: 7, 3: 532, 4: 167}\n",
      "Cluster 1, First Client 2: {0: 755, 6: 397, 7: 386}\n",
      "Cluster 2, First Client 3: {1: 68, 5: 1286, 8: 124}\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=0.6\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": 91,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Breaking HC\n"
     ]
    }
   ],
   "source": [
    "clusters = hierarchical_clustering(copy.deepcopy(adj_mat), thresh=args.cluster_alpha, linkage='average')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 92,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "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",
      "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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l3ECwMsvv4ybb74Z1157Lfr374+PPvrIvNm50XHOmzcPAHDDDTcgIiICn332Ge6880507doVd911Fz788ENkZ2f7RayOFBQUmBdN9evXDzExMYbGedVVVwEAtm/fjlOnThV5fseOHThx4gQiIyPRokULQ+LMzMyE+q9IRZUqVeyeU7VqVfP9VatWldimN3nj/Yn0wcROBzt27AAAREVF2fziWatVq5b5/vbt20ts8/Lly9izZw8AoHbt2g7P09p1pk1vxeoNRsSZn5+Pr776CoDlj3JJfBnnxYsXsWPHDjz33HN48803ER0djffff9/p13s71iFDhuDkyZN45513nE42jYgTACZNmoQff/wRS5YswfTp0zF48GCkpqZiwYIFhse5du1aAED58uXRokULPPjgg5g9ezaWLl2KWbNm4ZFHHkHdunXx559/Gh6rI4sXL8b+/fsBAA888IDTr/NWnPfccw+aNm2K7OxsdOvWDcuXL8e5c+dw+vRpLFy4ED169AAApKenO7yut+MsXbq0+f6hQ4fsnnPw4EHz/W3btpXYprd46/2J9MHETgenT58GACQnJztcPl6mTBnz/czMzBLbPHv2LAoKCoq81lG7zrTprVi9wYg433jjDWzZsgVhYWEYNWqU38RZpUoVmEwmxMbGol69enj11VfRq1cvrF69GjfddJPT7Xgz1rlz52L27Nno1q0b7r77bqdf5+s4r7/+enz55ZfYuXMnLly4gNOnT+Obb75B48aNkZmZiTvuuANLliwxNM4jR44AAD744ANs2bIFL7/8Mo4ePYoLFy5gyZIlqFevHg4fPoxu3bohIyPD0FgdmTRpEgCgefPmaNKkidOv81ackZGRWLlyJYYOHYrdu3ejU6dOSEpKQtmyZdGzZ0/ExcVhwYIFeO655wyLMz4+Ho0aNQIAh6VCrB/XYjCCt96fSB9M7HSgdclHRUU5PMd6KCInJ8fpNp1t15k2rdvVM1Zv8HWcixYtMv9Rf/7559G8eXOnXueLOFu1aoV27dqhSZMmSExMBAAsXboU06dPd6mWnbdiPXXqFB555BEkJCTgww8/dDoeR7z5Nf3pp5/Qv39/1KlTBzExMUhOTkavXr2wZs0aNGvWDHl5eXjssccMjTMrKwuA9Io8++yzeO6551ChQgXExMSgS5cuWLRoEaKjo3H8+HG89dZbhsZq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ccUVuPbaa/Hpp5/i0qVLDj+/gQMHwmQyoVOnTgCAjRs3on///qhSpQqio6NRpUoVDBw4ELt37y7xa3X27Fm89tpr6NChA6644grz69u3b49XX30Ve/bscfja9evX4/7770ft2rURHx+PhIQENG7cGKNGjbL5+jhL+76MGTMGALB///4iX0ftc7YnOzsbL730Eho2bIj4+HiULl0anTp1wpw5cxy+pvB8vs2bN+P+++9HjRo1EB0djdKlS+v2ebv6c6XX56jJysrCuHHjcPXVVyM5ORkxMTGoVq0a+vXrh99++63E15dk4cKFuO6661CmTBnEx8ejYcOGeOmll5CTk+Nx2wDwySefAABuu+02xMbGFnvukSNH8Pjjj6NWrVrmz/OWW27B2rVrzee88sorMJlMuPfee+220bFjR1SpUgUXL17EF198ocvnQKQrRRSgJk+erAAoAOrXX3916jXjx483v2bOnDk2z/3666/m5/bu3WvzXFZWlmrdurX5eUe32bNnm19TvXr1Es+3vs6TTz5Z4vmtW7dWmZmZdj+3AQMGKACqY8eOaubMmSo6OtpuG8nJyWrjxo0Ov0ZLlixR5cqVKzaOXr16FXldfn6+Gjp0qDKZTA5fV65cOfX777+X+H2yZv19cXTr2LGj3fPXrFmjrrzySoevGzt2rN1rpqenKwCqevXqav78+SomJsbmdUlJSbp83u78XOn1OSql1MaNG1XlypWLvfawYcNUQUFBkdfu3bu3xN+/oUOHOmy3UaNGat68eQ5/55xx/vx5FRkZqQCoGTNmFHvuL7/8ohITE+3GEhYWpn744QellFKNGzdWANS8efMctjVo0CAFQLVt29blmIm8jYkdBSx3ErsVK1aYX/P000/bPFdcYvf8888rACo8PFw999xzasOGDer48ePq4MGDas2aNeqjjz5SHTp0UHPnzjW/Jjs7W23ZssXc5g8//KCysrJsbtZvmM8//7zq3bu3mjZtmlq3bp06ePCgOn78uPrrr7/U6NGjVZkyZRQAddddd9n93LTErnLlyio6Olp16NBB/fzzz+rEiRPq4MGD6u2331ZRUVEKgGrTpo3dNv744w/zOcnJyWrcuHFq8+bN6vTp0+rAgQNq4cKF6oEHHlB9+/Yt8tphw4aZP9d7771XrVixQmVkZKhjx46p+fPnq4YNGyoAKiUlRR05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1Jw5c1THjh0VADVgwIAi17HeK7Y42h626enpdp//5ZdfVHJycrGfq729YgsKClR6eroKDw8v8Wt12223FRujI23btrXbnqO9YovbzaC4nSyc3eVCKc8+b3d/rvT4HJVybq/Yp59+2it7xTZs2NDjvWKVUqpnz54KgOrfv3+x573//vsO9/OtVKmSOnz4sOrTp4/N4/Pnzy/STkFBgapataoCoN588023YibyJvbYUdCJjo5GSkoK6tWrhz59+mDixInYu3cv5syZU2KNN0f+/PNPTJgwATfddBPS0tKQmJiIyMhIlC9fHtdffz0+/fRTrF27FuXLly/y2oceeghfffUVOnXqhDJlyiAszP6v3TPPPIPvvvsOXbt2RWJiIqKjo5GamopBgwZh3bp1uP32292K3VXXXnutuVelVatWSE5ORlRUFKpWrYr27dtj3LhxdldpmkwmjB49Gtu2bcNTTz2FJk2aICkpyTyM2KRJEzzyyCNYvHgxZs2a5VZsP/zwA5555hnUr19f9x4sd3nyeXvyc6WHRo0aYdu2bXjllVfQsmVLJCUlmb/Xd999N1atWoU33njDpWLe1iZOnIhvv/0WXbt2RenSpc092C+88AJ+//13l2pJOvLwww8DAObPn29T2qSwIUOGYPXq1bjzzjtRqVIlREREIC4uDu3atcPSpUtRqVIlfP755xg8eHCxK4CXL1+OgwcPIjo6ukjPNZE/MCnlg319iIiIvKCgoAA1a9bE/v37MX36dPTt29er13vwwQfx2WefoW/fvh7VriTyFvbYERFRwAoLCzPv0/vee+959VonT57EzJkzER4ebt7/l8jfMLEjIqKA9sADD6BevXr4/fffsWjRIq9d5/XXX8f58+dx3333oWHDhl67DpEnOBRLREQB79tvv0WvXr3QunVrrFmzRvf2T548iRo1akAphV27drm0JSGRLzGxIyIiIgoSHIolIiIiChJM7IiIiIiCBBM7IiIioiDBxI6IiIgoSDCxIyIiIgoSTOyIiIiIggQTOyIiIqIgwcSOiIiIKEgwsSMiIiIKEkzsiIiIiIIEEzsiIiKiIPH/miGDPyFSLk4AAAAASUVORK5CYII=",
      "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": 6,
   "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",
    "# Robust FedAvg over a list of encoders (any architecture)\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": 7,
   "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": 8,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "##### ROUND 1 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.5052\n",
      "avg best PRE   : 2.00%\n",
      "avg best POST  : 15.29%\n",
      "avg best ANY   : 15.29%\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= 62.00 | best_any= 62.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=  0.00 | best_any=  0.00\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= 48.10 | best_any= 48.10\n",
      "Client  16 | best_pre=  0.00 | best_post= 70.93 | best_any= 70.93\n",
      "Client  17 | best_pre= 33.33 | best_post= 66.23 | best_any= 66.23\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=  0.00 | best_any=  0.00\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=  0.00 | best_post= 91.70 | best_any= 91.70\n",
      "Client  27 | best_pre=  0.00 | best_post= 92.10 | best_any= 92.10\n",
      "Client  28 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  29 | best_pre= 33.33 | best_post= 91.30 | best_any= 91.30\n",
      "Client  30 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\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= 33.33 | best_post= 93.20 | best_any= 93.20\n",
      "Client  34 | best_pre= 33.33 | best_post= 69.70 | best_any= 69.70\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= 60.97 | best_any= 60.97\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=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  44 | best_pre=  0.00 | best_post= 78.03 | best_any= 78.03\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= 71.20 | best_any= 71.20\n",
      "Client  50 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  51 | best_pre=  0.00 | best_post= 96.40 | best_any= 96.40\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=  0.00 | best_any=  0.00\n",
      "Client  56 | best_pre=  0.00 | best_post= 67.07 | best_any= 67.07\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= 33.33 | best_post= 96.73 | best_any= 96.73\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=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  65 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  66 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  67 | best_pre= 33.33 | best_post= 66.30 | best_any= 66.30\n",
      "Client  68 | best_pre=  0.00 | best_post= 63.10 | best_any= 63.10\n",
      "Client  69 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\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=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  76 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\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=  0.00 | best_post= 56.53 | best_any= 56.53\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= 95.83 | best_any= 95.83\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=  0.00 | best_any=  0.00\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= 91.90 | best_any= 91.90\n",
      "Client  93 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  94 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\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=  0.00 | best_any=  0.00\n",
      "\n",
      "##### ROUND 2 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.2306\n",
      "avg best PRE   : 16.88%\n",
      "avg best POST  : 28.84%\n",
      "avg best ANY   : 29.05%\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= 91.07 | best_post= 88.70 | best_any= 91.07\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= 62.00 | best_any= 62.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=  0.00 | best_any=  0.00\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= 48.10 | best_any= 48.10\n",
      "Client  16 | best_pre=  0.00 | best_post= 70.93 | best_any= 70.93\n",
      "Client  17 | best_pre= 33.33 | best_post= 66.23 | best_any= 66.23\n",
      "Client  18 | best_pre= 33.33 | best_post= 62.63 | best_any= 62.63\n",
      "Client  19 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  20 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  21 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  22 | best_pre= 33.33 | best_post= 73.90 | best_any= 73.90\n",
      "Client  23 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  24 | best_pre= 93.70 | best_post= 96.27 | best_any= 96.27\n",
      "Client  25 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  26 | best_pre= 82.00 | best_post= 93.43 | best_any= 93.43\n",
      "Client  27 | best_pre=  0.00 | best_post= 92.10 | best_any= 92.10\n",
      "Client  28 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  29 | best_pre= 93.70 | best_post= 94.30 | best_any= 94.30\n",
      "Client  30 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  31 | best_pre= 82.00 | best_post= 92.63 | best_any= 92.63\n",
      "Client  32 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  33 | best_pre= 33.33 | best_post= 93.20 | best_any= 93.20\n",
      "Client  34 | best_pre= 93.70 | best_post= 84.53 | best_any= 93.70\n",
      "Client  35 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  36 | best_pre= 93.70 | best_post= 90.43 | best_any= 93.70\n",
      "Client  37 | best_pre= 91.07 | best_post= 96.00 | best_any= 96.00\n",
      "Client  38 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  39 | best_pre= 91.07 | best_post= 96.93 | best_any= 96.93\n",
      "Client  40 | best_pre=  0.00 | best_post= 60.97 | best_any= 60.97\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=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  44 | best_pre=  0.00 | best_post= 78.03 | best_any= 78.03\n",
      "Client  45 | best_pre= 93.70 | best_post= 97.47 | best_any= 97.47\n",
      "Client  46 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  47 | best_pre= 91.07 | best_post= 96.73 | best_any= 96.73\n",
      "Client  48 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  49 | best_pre=  0.00 | best_post= 71.20 | best_any= 71.20\n",
      "Client  50 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  51 | best_pre=  0.00 | best_post= 96.40 | best_any= 96.40\n",
      "Client  52 | best_pre= 33.33 | best_post= 53.33 | best_any= 53.33\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=  0.00 | best_any=  0.00\n",
      "Client  56 | best_pre=  0.00 | best_post= 67.07 | best_any= 67.07\n",
      "Client  57 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  58 | best_pre= 91.07 | best_post= 95.00 | best_any= 95.00\n",
      "Client  59 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  60 | best_pre= 33.33 | best_post= 96.73 | best_any= 96.73\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=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  65 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  66 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  67 | best_pre= 93.70 | best_post= 87.73 | best_any= 93.70\n",
      "Client  68 | best_pre=  0.00 | best_post= 63.10 | best_any= 63.10\n",
      "Client  69 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\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=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  76 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  77 | best_pre= 91.07 | best_post= 96.17 | best_any= 96.17\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=  0.00 | best_post= 56.53 | best_any= 56.53\n",
      "Client  81 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  82 | best_pre= 33.33 | best_post= 81.10 | best_any= 81.10\n",
      "Client  83 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  84 | best_pre= 91.07 | best_post= 96.20 | best_any= 96.20\n",
      "Client  85 | best_pre= 91.07 | best_post= 96.47 | best_any= 96.47\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=  0.00 | best_any=  0.00\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= 91.90 | best_any= 91.90\n",
      "Client  93 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  94 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\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=  0.00 | best_any=  0.00\n",
      "\n",
      "##### ROUND 3 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.7921\n",
      "avg best PRE   : 30.86%\n",
      "avg best POST  : 42.05%\n",
      "avg best ANY   : 42.85%\n",
      "\n",
      "Client   0 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   1 | best_pre= 97.23 | best_post= 91.97 | best_any= 97.23\n",
      "Client   2 | best_pre= 97.27 | best_post= 88.70 | best_any= 97.27\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= 62.00 | best_any= 62.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= 97.23 | best_post= 95.20 | best_any= 97.23\n",
      "Client  12 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\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= 48.10 | best_any= 48.10\n",
      "Client  16 | best_pre=  0.00 | best_post= 70.93 | best_any= 70.93\n",
      "Client  17 | best_pre= 33.33 | best_post= 66.23 | best_any= 66.23\n",
      "Client  18 | best_pre= 33.33 | best_post= 62.63 | best_any= 62.63\n",
      "Client  19 | best_pre= 97.27 | best_post= 95.73 | best_any= 97.27\n",
      "Client  20 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  21 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  22 | best_pre= 33.33 | best_post= 73.90 | best_any= 73.90\n",
      "Client  23 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  24 | best_pre= 93.70 | best_post= 96.27 | best_any= 96.27\n",
      "Client  25 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  26 | best_pre= 82.00 | best_post= 93.43 | best_any= 93.43\n",
      "Client  27 | best_pre=  0.00 | best_post= 92.10 | best_any= 92.10\n",
      "Client  28 | best_pre= 53.67 | best_post= 46.43 | best_any= 53.67\n",
      "Client  29 | best_pre= 93.70 | best_post= 94.30 | best_any= 94.30\n",
      "Client  30 | best_pre= 97.27 | best_post= 97.23 | best_any= 97.27\n",
      "Client  31 | best_pre= 91.60 | best_post= 92.63 | best_any= 92.63\n",
      "Client  32 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  33 | best_pre= 33.33 | best_post= 93.20 | best_any= 93.20\n",
      "Client  34 | best_pre= 93.70 | best_post= 84.53 | best_any= 93.70\n",
      "Client  35 | best_pre= 53.67 | best_post= 73.67 | best_any= 73.67\n",
      "Client  36 | best_pre= 93.70 | best_post= 90.43 | best_any= 93.70\n",
      "Client  37 | best_pre= 97.27 | best_post= 97.73 | best_any= 97.73\n",
      "Client  38 | best_pre= 97.27 | best_post= 97.60 | best_any= 97.60\n",
      "Client  39 | best_pre= 91.07 | best_post= 96.93 | best_any= 96.93\n",
      "Client  40 | best_pre=  0.00 | best_post= 60.97 | best_any= 60.97\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= 91.60 | best_post= 93.80 | best_any= 93.80\n",
      "Client  44 | best_pre=  0.00 | best_post= 78.03 | best_any= 78.03\n",
      "Client  45 | best_pre= 93.70 | best_post= 97.47 | best_any= 97.47\n",
      "Client  46 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  47 | best_pre= 91.07 | best_post= 96.73 | best_any= 96.73\n",
      "Client  48 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  49 | best_pre=  0.00 | best_post= 71.20 | best_any= 71.20\n",
      "Client  50 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  51 | best_pre=  0.00 | best_post= 96.40 | best_any= 96.40\n",
      "Client  52 | best_pre= 33.33 | best_post= 53.33 | best_any= 53.33\n",
      "Client  53 | best_pre= 53.67 | best_post= 57.13 | best_any= 57.13\n",
      "Client  54 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  55 | best_pre= 53.67 | best_post= 60.97 | best_any= 60.97\n",
      "Client  56 | best_pre=  0.00 | best_post= 67.07 | best_any= 67.07\n",
      "Client  57 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  58 | best_pre= 91.07 | best_post= 95.00 | best_any= 95.00\n",
      "Client  59 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  60 | best_pre= 33.33 | best_post= 96.73 | best_any= 96.73\n",
      "Client  61 | best_pre= 81.23 | best_post= 67.50 | best_any= 81.23\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=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  65 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  66 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  67 | best_pre= 93.70 | best_post= 87.73 | best_any= 93.70\n",
      "Client  68 | best_pre=  0.00 | best_post= 63.10 | best_any= 63.10\n",
      "Client  69 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\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=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  76 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  77 | best_pre= 91.07 | best_post= 96.17 | best_any= 96.17\n",
      "Client  78 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  79 | best_pre= 81.23 | best_post= 85.27 | best_any= 85.27\n",
      "Client  80 | best_pre= 53.67 | best_post= 63.67 | best_any= 63.67\n",
      "Client  81 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  82 | best_pre= 33.33 | best_post= 81.10 | best_any= 81.10\n",
      "Client  83 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  84 | best_pre= 91.07 | best_post= 96.20 | best_any= 96.20\n",
      "Client  85 | best_pre= 91.07 | best_post= 96.47 | best_any= 96.47\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=  0.00 | best_any=  0.00\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= 91.60 | best_post= 83.80 | best_any= 91.60\n",
      "Client  92 | best_pre=  0.00 | best_post= 91.90 | best_any= 91.90\n",
      "Client  93 | best_pre= 97.23 | best_post= 83.13 | best_any= 97.23\n",
      "Client  94 | best_pre= 81.23 | best_post= 86.70 | best_any= 86.70\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= 97.27 | best_post= 95.37 | best_any= 97.27\n",
      "\n",
      "##### ROUND 4 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.5180\n",
      "avg best PRE   : 41.80%\n",
      "avg best POST  : 51.39%\n",
      "avg best ANY   : 52.70%\n",
      "\n",
      "Client   0 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   1 | best_pre= 97.23 | best_post= 91.97 | best_any= 97.23\n",
      "Client   2 | best_pre= 97.27 | best_post= 88.70 | best_any= 97.27\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= 62.00 | best_any= 62.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= 90.90 | best_post= 93.30 | best_any= 93.30\n",
      "Client  10 | best_pre= 93.50 | best_post= 96.97 | best_any= 96.97\n",
      "Client  11 | best_pre= 97.23 | best_post= 95.20 | best_any= 97.23\n",
      "Client  12 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  13 | best_pre= 61.50 | best_post= 46.17 | best_any= 61.50\n",
      "Client  14 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  15 | best_pre=  0.00 | best_post= 48.10 | best_any= 48.10\n",
      "Client  16 | best_pre= 90.90 | best_post= 92.37 | best_any= 92.37\n",
      "Client  17 | best_pre= 33.33 | best_post= 66.23 | best_any= 66.23\n",
      "Client  18 | best_pre= 33.33 | best_post= 62.63 | best_any= 62.63\n",
      "Client  19 | best_pre= 97.27 | best_post= 95.73 | best_any= 97.27\n",
      "Client  20 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  21 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  22 | best_pre= 33.33 | best_post= 73.90 | best_any= 73.90\n",
      "Client  23 | best_pre= 97.73 | best_post= 97.17 | best_any= 97.73\n",
      "Client  24 | best_pre= 93.70 | best_post= 97.30 | best_any= 97.30\n",
      "Client  25 | best_pre= 93.50 | best_post= 91.33 | best_any= 93.50\n",
      "Client  26 | best_pre= 82.00 | best_post= 93.43 | best_any= 93.43\n",
      "Client  27 | best_pre=  0.00 | best_post= 92.10 | best_any= 92.10\n",
      "Client  28 | best_pre= 61.50 | best_post= 48.60 | best_any= 61.50\n",
      "Client  29 | best_pre= 93.70 | best_post= 94.30 | best_any= 94.30\n",
      "Client  30 | best_pre= 97.27 | best_post= 97.23 | best_any= 97.27\n",
      "Client  31 | best_pre= 91.60 | best_post= 92.63 | best_any= 92.63\n",
      "Client  32 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  33 | best_pre= 33.33 | best_post= 93.20 | best_any= 93.20\n",
      "Client  34 | best_pre= 93.70 | best_post= 84.53 | best_any= 93.70\n",
      "Client  35 | best_pre= 53.67 | best_post= 73.67 | best_any= 73.67\n",
      "Client  36 | best_pre= 93.70 | best_post= 90.43 | best_any= 93.70\n",
      "Client  37 | best_pre= 97.27 | best_post= 97.73 | best_any= 97.73\n",
      "Client  38 | best_pre= 97.27 | best_post= 97.60 | best_any= 97.60\n",
      "Client  39 | best_pre= 97.73 | best_post= 97.87 | best_any= 97.87\n",
      "Client  40 | best_pre=  0.00 | best_post= 60.97 | best_any= 60.97\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= 91.60 | best_post= 93.80 | best_any= 93.80\n",
      "Client  44 | best_pre= 65.73 | best_post= 86.77 | best_any= 86.77\n",
      "Client  45 | best_pre= 93.70 | best_post= 97.47 | best_any= 97.47\n",
      "Client  46 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  47 | best_pre= 91.07 | best_post= 96.73 | best_any= 96.73\n",
      "Client  48 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  49 | best_pre=  0.00 | best_post= 71.20 | best_any= 71.20\n",
      "Client  50 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  51 | best_pre=  0.00 | best_post= 96.40 | best_any= 96.40\n",
      "Client  52 | best_pre= 33.33 | best_post= 53.33 | best_any= 53.33\n",
      "Client  53 | best_pre= 53.67 | best_post= 57.13 | best_any= 57.13\n",
      "Client  54 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  55 | best_pre= 61.50 | best_post= 60.97 | best_any= 61.50\n",
      "Client  56 | best_pre=  0.00 | best_post= 67.07 | best_any= 67.07\n",
      "Client  57 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  58 | best_pre= 91.07 | best_post= 95.00 | best_any= 95.00\n",
      "Client  59 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  60 | best_pre= 33.33 | best_post= 96.73 | best_any= 96.73\n",
      "Client  61 | best_pre= 81.23 | best_post= 67.50 | best_any= 81.23\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=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  65 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  66 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  67 | best_pre= 93.70 | best_post= 87.73 | best_any= 93.70\n",
      "Client  68 | best_pre=  0.00 | best_post= 63.10 | best_any= 63.10\n",
      "Client  69 | best_pre= 93.50 | best_post= 72.00 | best_any= 93.50\n",
      "Client  70 | best_pre= 97.73 | best_post= 96.57 | best_any= 97.73\n",
      "Client  71 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  72 | best_pre= 61.50 | best_post= 64.80 | best_any= 64.80\n",
      "Client  73 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  74 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  76 | best_pre= 90.90 | best_post= 89.47 | best_any= 90.90\n",
      "Client  77 | best_pre= 91.07 | best_post= 96.17 | best_any= 96.17\n",
      "Client  78 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  79 | best_pre= 81.23 | best_post= 85.27 | best_any= 85.27\n",
      "Client  80 | best_pre= 53.67 | best_post= 63.67 | best_any= 63.67\n",
      "Client  81 | best_pre= 61.50 | best_post= 69.50 | best_any= 69.50\n",
      "Client  82 | best_pre= 33.33 | best_post= 81.10 | best_any= 81.10\n",
      "Client  83 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  84 | best_pre= 97.73 | best_post= 96.27 | best_any= 97.73\n",
      "Client  85 | best_pre= 91.07 | best_post= 96.47 | best_any= 96.47\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=  0.00 | best_any=  0.00\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= 91.60 | best_post= 83.80 | best_any= 91.60\n",
      "Client  92 | best_pre=  0.00 | best_post= 91.90 | best_any= 91.90\n",
      "Client  93 | best_pre= 97.23 | best_post= 83.13 | best_any= 97.23\n",
      "Client  94 | best_pre= 81.23 | best_post= 86.70 | best_any= 86.70\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= 65.73 | best_post= 82.30 | best_any= 82.30\n",
      "Client  99 | best_pre= 97.73 | best_post= 95.37 | best_any= 97.73\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.1322\n",
      "avg best PRE   : 80.48%\n",
      "avg best POST  : 81.84%\n",
      "avg best ANY   : 84.04%\n",
      "\n",
      "Client   0 | best_pre= 97.57 | best_post= 97.83 | best_any= 97.83\n",
      "Client   1 | best_pre= 97.23 | best_post= 91.97 | best_any= 97.23\n",
      "Client   2 | best_pre= 97.27 | best_post= 88.70 | best_any= 97.27\n",
      "Client   3 | best_pre= 77.90 | best_post= 79.03 | best_any= 79.03\n",
      "Client   4 | best_pre= 92.20 | best_post= 92.20 | best_any= 92.20\n",
      "Client   5 | best_pre= 81.63 | best_post= 81.23 | best_any= 81.63\n",
      "Client   6 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   7 | best_pre= 98.00 | best_post= 97.83 | best_any= 98.00\n",
      "Client   8 | best_pre= 97.57 | best_post= 97.70 | best_any= 97.70\n",
      "Client   9 | best_pre= 93.70 | best_post= 93.30 | best_any= 93.70\n",
      "Client  10 | best_pre= 96.40 | best_post= 97.73 | best_any= 97.73\n",
      "Client  11 | best_pre= 97.27 | best_post= 97.73 | best_any= 97.73\n",
      "Client  12 | best_pre= 81.63 | best_post= 77.27 | best_any= 81.63\n",
      "Client  13 | best_pre= 61.50 | best_post= 46.17 | best_any= 61.50\n",
      "Client  14 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  15 | best_pre= 81.63 | best_post= 79.47 | best_any= 81.63\n",
      "Client  16 | best_pre= 92.20 | best_post= 93.27 | best_any= 93.27\n",
      "Client  17 | best_pre= 98.00 | best_post= 87.93 | best_any= 98.00\n",
      "Client  18 | best_pre= 76.40 | best_post= 74.20 | best_any= 76.40\n",
      "Client  19 | best_pre= 97.27 | best_post= 95.73 | best_any= 97.27\n",
      "Client  20 | best_pre= 98.00 | best_post= 97.73 | best_any= 98.00\n",
      "Client  21 | best_pre= 97.60 | best_post= 98.07 | best_any= 98.07\n",
      "Client  22 | best_pre= 82.17 | best_post= 87.77 | best_any= 87.77\n",
      "Client  23 | best_pre= 98.57 | best_post= 97.57 | best_any= 98.57\n",
      "Client  24 | best_pre= 98.00 | best_post= 97.83 | best_any= 98.00\n",
      "Client  25 | best_pre= 93.50 | best_post= 91.33 | best_any= 93.50\n",
      "Client  26 | best_pre= 82.00 | best_post= 93.43 | best_any= 93.43\n",
      "Client  27 | best_pre= 98.33 | best_post= 97.03 | best_any= 98.33\n",
      "Client  28 | best_pre= 81.63 | best_post= 78.83 | best_any= 81.63\n",
      "Client  29 | best_pre= 93.70 | best_post= 94.30 | best_any= 94.30\n",
      "Client  30 | best_pre= 98.50 | best_post= 98.37 | best_any= 98.50\n",
      "Client  31 | best_pre= 94.17 | best_post= 92.97 | best_any= 94.17\n",
      "Client  32 | best_pre= 82.17 | best_post= 87.77 | best_any= 87.77\n",
      "Client  33 | best_pre= 97.57 | best_post= 96.47 | best_any= 97.57\n",
      "Client  34 | best_pre= 98.00 | best_post= 95.83 | best_any= 98.00\n",
      "Client  35 | best_pre= 81.13 | best_post= 75.87 | best_any= 81.13\n",
      "Client  36 | best_pre= 97.27 | best_post= 97.30 | best_any= 97.30\n",
      "Client  37 | best_pre= 97.67 | best_post= 98.03 | best_any= 98.03\n",
      "Client  38 | best_pre= 98.57 | best_post= 98.13 | best_any= 98.57\n",
      "Client  39 | best_pre= 98.33 | best_post= 98.13 | best_any= 98.33\n",
      "Client  40 | best_pre= 71.63 | best_post= 88.50 | best_any= 88.50\n",
      "Client  41 | best_pre= 97.27 | best_post= 97.60 | best_any= 97.60\n",
      "Client  42 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  43 | best_pre= 94.17 | best_post= 93.80 | best_any= 94.17\n",
      "Client  44 | best_pre= 68.37 | best_post= 90.43 | best_any= 90.43\n",
      "Client  45 | best_pre= 98.00 | best_post= 98.13 | best_any= 98.13\n",
      "Client  46 | best_pre= 94.10 | best_post= 93.97 | best_any= 94.10\n",
      "Client  47 | best_pre= 98.37 | best_post= 97.87 | best_any= 98.37\n",
      "Client  48 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  49 | best_pre= 93.97 | best_post= 92.33 | best_any= 93.97\n",
      "Client  50 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  51 | best_pre=  0.00 | best_post= 96.40 | best_any= 96.40\n",
      "Client  52 | best_pre= 81.37 | best_post= 71.10 | best_any= 81.37\n",
      "Client  53 | best_pre= 81.63 | best_post= 70.40 | best_any= 81.63\n",
      "Client  54 | best_pre= 97.50 | best_post= 94.83 | best_any= 97.50\n",
      "Client  55 | best_pre= 81.63 | best_post= 74.60 | best_any= 81.63\n",
      "Client  56 | best_pre= 94.10 | best_post= 89.13 | best_any= 94.10\n",
      "Client  57 | best_pre= 97.57 | best_post= 97.70 | best_any= 97.70\n",
      "Client  58 | best_pre= 97.67 | best_post= 96.67 | best_any= 97.67\n",
      "Client  59 | best_pre= 71.63 | best_post= 87.70 | best_any= 87.70\n",
      "Client  60 | best_pre= 98.00 | best_post= 98.07 | best_any= 98.07\n",
      "Client  61 | best_pre= 82.17 | best_post= 87.67 | best_any= 87.67\n",
      "Client  62 | best_pre= 96.40 | best_post= 97.87 | best_any= 97.87\n",
      "Client  63 | best_pre= 97.50 | best_post= 93.67 | best_any= 97.50\n",
      "Client  64 | best_pre= 71.63 | best_post= 86.90 | best_any= 86.90\n",
      "Client  65 | best_pre= 66.70 | best_post= 84.13 | best_any= 84.13\n",
      "Client  66 | best_pre= 81.37 | best_post= 46.73 | best_any= 81.37\n",
      "Client  67 | best_pre= 97.57 | best_post= 89.87 | best_any= 97.57\n",
      "Client  68 | best_pre= 45.20 | best_post= 86.07 | best_any= 86.07\n",
      "Client  69 | best_pre= 98.00 | best_post= 98.03 | best_any= 98.03\n",
      "Client  70 | best_pre= 97.73 | best_post= 97.33 | best_any= 97.73\n",
      "Client  71 | best_pre= 81.13 | best_post= 73.40 | best_any= 81.13\n",
      "Client  72 | best_pre= 61.50 | best_post= 64.80 | best_any= 64.80\n",
      "Client  73 | best_pre= 80.53 | best_post= 72.07 | best_any= 80.53\n",
      "Client  74 | best_pre= 96.40 | best_post= 92.93 | best_any= 96.40\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  76 | best_pre= 93.97 | best_post= 92.07 | best_any= 93.97\n",
      "Client  77 | best_pre= 98.37 | best_post= 98.03 | best_any= 98.37\n",
      "Client  78 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  79 | best_pre= 81.23 | best_post= 85.27 | best_any= 85.27\n",
      "Client  80 | best_pre= 53.67 | best_post= 63.67 | best_any= 63.67\n",
      "Client  81 | best_pre= 81.63 | best_post= 78.50 | best_any= 81.63\n",
      "Client  82 | best_pre= 33.33 | best_post= 81.10 | best_any= 81.10\n",
      "Client  83 | best_pre= 73.67 | best_post= 75.00 | best_any= 75.00\n",
      "Client  84 | best_pre= 98.50 | best_post= 97.63 | best_any= 98.50\n",
      "Client  85 | best_pre= 98.33 | best_post= 96.90 | best_any= 98.33\n",
      "Client  86 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  87 | best_pre= 93.70 | best_post= 92.60 | best_any= 93.70\n",
      "Client  88 | best_pre= 94.17 | best_post= 94.17 | best_any= 94.17\n",
      "Client  89 | best_pre= 98.57 | best_post= 98.17 | best_any= 98.57\n",
      "Client  90 | best_pre= 68.37 | best_post= 87.83 | best_any= 87.83\n",
      "Client  91 | best_pre= 94.10 | best_post= 87.53 | best_any= 94.10\n",
      "Client  92 | best_pre= 97.67 | best_post= 94.83 | best_any= 97.67\n",
      "Client  93 | best_pre= 97.23 | best_post= 83.90 | best_any= 97.23\n",
      "Client  94 | best_pre= 81.23 | best_post= 90.07 | best_any= 90.07\n",
      "Client  95 | best_pre= 98.00 | best_post= 97.17 | best_any= 98.00\n",
      "Client  96 | best_pre= 97.57 | best_post= 96.97 | best_any= 97.57\n",
      "Client  97 | best_pre= 81.37 | best_post= 70.37 | best_any= 81.37\n",
      "Client  98 | best_pre= 82.17 | best_post= 82.30 | best_any= 82.30\n",
      "Client  99 | best_pre= 98.57 | best_post= 97.43 | best_any= 98.57\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.0634\n",
      "avg best PRE   : 89.82%\n",
      "avg best POST  : 89.23%\n",
      "avg best ANY   : 91.24%\n",
      "\n",
      "Client   0 | best_pre= 97.90 | best_post= 98.33 | best_any= 98.33\n",
      "Client   1 | best_pre= 97.23 | best_post= 91.97 | best_any= 97.23\n",
      "Client   2 | best_pre= 98.40 | best_post= 96.87 | best_any= 98.40\n",
      "Client   3 | best_pre= 83.73 | best_post= 80.33 | best_any= 83.73\n",
      "Client   4 | best_pre= 94.93 | best_post= 93.13 | best_any= 94.93\n",
      "Client   5 | best_pre= 82.33 | best_post= 81.90 | best_any= 82.33\n",
      "Client   6 | best_pre= 82.37 | best_post= 83.07 | best_any= 83.07\n",
      "Client   7 | best_pre= 98.40 | best_post= 98.33 | best_any= 98.40\n",
      "Client   8 | best_pre= 98.17 | best_post= 98.03 | best_any= 98.17\n",
      "Client   9 | best_pre= 93.70 | best_post= 93.30 | best_any= 93.70\n",
      "Client  10 | best_pre= 98.43 | best_post= 98.30 | best_any= 98.43\n",
      "Client  11 | best_pre= 97.90 | best_post= 98.13 | best_any= 98.13\n",
      "Client  12 | best_pre= 82.40 | best_post= 79.23 | best_any= 82.40\n",
      "Client  13 | best_pre= 83.37 | best_post= 70.63 | best_any= 83.37\n",
      "Client  14 | best_pre= 93.63 | best_post= 90.70 | best_any= 93.63\n",
      "Client  15 | best_pre= 81.63 | best_post= 79.47 | best_any= 81.63\n",
      "Client  16 | best_pre= 94.93 | best_post= 95.07 | best_any= 95.07\n",
      "Client  17 | best_pre= 98.40 | best_post= 90.73 | best_any= 98.40\n",
      "Client  18 | best_pre= 82.67 | best_post= 74.20 | best_any= 82.67\n",
      "Client  19 | best_pre= 98.53 | best_post= 97.73 | best_any= 98.53\n",
      "Client  20 | best_pre= 98.20 | best_post= 98.27 | best_any= 98.27\n",
      "Client  21 | best_pre= 98.40 | best_post= 98.43 | best_any= 98.43\n",
      "Client  22 | best_pre= 82.17 | best_post= 90.20 | best_any= 90.20\n",
      "Client  23 | best_pre= 98.70 | best_post= 98.10 | best_any= 98.70\n",
      "Client  24 | best_pre= 98.43 | best_post= 98.30 | best_any= 98.43\n",
      "Client  25 | best_pre= 96.73 | best_post= 98.03 | best_any= 98.03\n",
      "Client  26 | best_pre= 94.93 | best_post= 95.33 | best_any= 95.33\n",
      "Client  27 | best_pre= 98.37 | best_post= 97.93 | best_any= 98.37\n",
      "Client  28 | best_pre= 83.37 | best_post= 79.93 | best_any= 83.37\n",
      "Client  29 | best_pre= 98.27 | best_post= 94.43 | best_any= 98.27\n",
      "Client  30 | best_pre= 98.50 | best_post= 98.57 | best_any= 98.57\n",
      "Client  31 | best_pre= 94.93 | best_post= 93.87 | best_any= 94.93\n",
      "Client  32 | best_pre= 82.67 | best_post= 90.03 | best_any= 90.03\n",
      "Client  33 | best_pre= 98.40 | best_post= 96.47 | best_any= 98.40\n",
      "Client  34 | best_pre= 98.07 | best_post= 96.80 | best_any= 98.07\n",
      "Client  35 | best_pre= 81.13 | best_post= 81.17 | best_any= 81.17\n",
      "Client  36 | best_pre= 97.27 | best_post= 97.30 | best_any= 97.30\n",
      "Client  37 | best_pre= 98.37 | best_post= 98.33 | best_any= 98.37\n",
      "Client  38 | best_pre= 98.70 | best_post= 98.20 | best_any= 98.70\n",
      "Client  39 | best_pre= 98.37 | best_post= 98.37 | best_any= 98.37\n",
      "Client  40 | best_pre= 76.87 | best_post= 89.23 | best_any= 89.23\n",
      "Client  41 | best_pre= 98.13 | best_post= 98.27 | best_any= 98.27\n",
      "Client  42 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  43 | best_pre= 94.67 | best_post= 94.37 | best_any= 94.67\n",
      "Client  44 | best_pre= 87.80 | best_post= 91.33 | best_any= 91.33\n",
      "Client  45 | best_pre= 98.40 | best_post= 98.17 | best_any= 98.40\n",
      "Client  46 | best_pre= 94.67 | best_post= 94.67 | best_any= 94.67\n",
      "Client  47 | best_pre= 98.53 | best_post= 98.17 | best_any= 98.53\n",
      "Client  48 | best_pre= 93.77 | best_post= 91.63 | best_any= 93.77\n",
      "Client  49 | best_pre= 94.07 | best_post= 92.50 | best_any= 94.07\n",
      "Client  50 | best_pre= 77.97 | best_post= 87.03 | best_any= 87.03\n",
      "Client  51 | best_pre= 98.40 | best_post= 97.83 | best_any= 98.40\n",
      "Client  52 | best_pre= 82.37 | best_post= 72.90 | best_any= 82.37\n",
      "Client  53 | best_pre= 82.37 | best_post= 71.30 | best_any= 82.37\n",
      "Client  54 | best_pre= 97.50 | best_post= 94.83 | best_any= 97.50\n",
      "Client  55 | best_pre= 82.47 | best_post= 74.93 | best_any= 82.47\n",
      "Client  56 | best_pre= 94.67 | best_post= 92.50 | best_any= 94.67\n",
      "Client  57 | best_pre= 98.40 | best_post= 98.30 | best_any= 98.40\n",
      "Client  58 | best_pre= 98.23 | best_post= 96.67 | best_any= 98.23\n",
      "Client  59 | best_pre= 71.63 | best_post= 87.70 | best_any= 87.70\n",
      "Client  60 | best_pre= 98.27 | best_post= 98.33 | best_any= 98.33\n",
      "Client  61 | best_pre= 82.17 | best_post= 88.23 | best_any= 88.23\n",
      "Client  62 | best_pre= 98.43 | best_post= 98.13 | best_any= 98.43\n",
      "Client  63 | best_pre= 98.43 | best_post= 95.67 | best_any= 98.43\n",
      "Client  64 | best_pre= 78.03 | best_post= 87.13 | best_any= 87.13\n",
      "Client  65 | best_pre= 83.07 | best_post= 86.53 | best_any= 86.53\n",
      "Client  66 | best_pre= 83.37 | best_post= 60.73 | best_any= 83.37\n",
      "Client  67 | best_pre= 98.43 | best_post= 91.13 | best_any= 98.43\n",
      "Client  68 | best_pre= 64.57 | best_post= 86.47 | best_any= 86.47\n",
      "Client  69 | best_pre= 98.00 | best_post= 98.03 | best_any= 98.03\n",
      "Client  70 | best_pre= 97.73 | best_post= 97.33 | best_any= 97.73\n",
      "Client  71 | best_pre= 82.40 | best_post= 73.40 | best_any= 82.40\n",
      "Client  72 | best_pre= 82.47 | best_post= 82.00 | best_any= 82.47\n",
      "Client  73 | best_pre= 83.37 | best_post= 72.07 | best_any= 83.37\n",
      "Client  74 | best_pre= 98.27 | best_post= 93.33 | best_any= 98.27\n",
      "Client  75 | best_pre= 98.43 | best_post= 98.47 | best_any= 98.47\n",
      "Client  76 | best_pre= 94.67 | best_post= 93.20 | best_any= 94.67\n",
      "Client  77 | best_pre= 98.70 | best_post= 98.37 | best_any= 98.70\n",
      "Client  78 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  79 | best_pre= 81.23 | best_post= 89.90 | best_any= 89.90\n",
      "Client  80 | best_pre= 83.37 | best_post= 80.53 | best_any= 83.37\n",
      "Client  81 | best_pre= 81.63 | best_post= 78.50 | best_any= 81.63\n",
      "Client  82 | best_pre= 74.77 | best_post= 89.73 | best_any= 89.73\n",
      "Client  83 | best_pre= 83.73 | best_post= 81.23 | best_any= 83.73\n",
      "Client  84 | best_pre= 98.50 | best_post= 97.63 | best_any= 98.50\n",
      "Client  85 | best_pre= 98.70 | best_post= 98.23 | best_any= 98.70\n",
      "Client  86 | best_pre= 98.63 | best_post= 98.20 | best_any= 98.63\n",
      "Client  87 | best_pre= 93.90 | best_post= 92.60 | best_any= 93.90\n",
      "Client  88 | best_pre= 94.93 | best_post= 95.03 | best_any= 95.03\n",
      "Client  89 | best_pre= 98.57 | best_post= 98.17 | best_any= 98.57\n",
      "Client  90 | best_pre= 83.07 | best_post= 91.27 | best_any= 91.27\n",
      "Client  91 | best_pre= 94.67 | best_post= 89.30 | best_any= 94.67\n",
      "Client  92 | best_pre= 98.63 | best_post= 98.03 | best_any= 98.63\n",
      "Client  93 | best_pre= 98.13 | best_post= 91.37 | best_any= 98.13\n",
      "Client  94 | best_pre= 81.63 | best_post= 90.90 | best_any= 90.90\n",
      "Client  95 | best_pre= 98.13 | best_post= 97.17 | best_any= 98.13\n",
      "Client  96 | best_pre= 98.13 | best_post= 98.03 | best_any= 98.13\n",
      "Client  97 | best_pre= 82.40 | best_post= 75.67 | best_any= 82.40\n",
      "Client  98 | best_pre= 87.80 | best_post= 85.23 | best_any= 87.80\n",
      "Client  99 | best_pre= 98.70 | best_post= 98.00 | best_any= 98.70\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.0381\n",
      "avg best PRE   : 92.79%\n",
      "avg best POST  : 91.77%\n",
      "avg best ANY   : 93.73%\n",
      "\n",
      "Client   0 | best_pre= 98.60 | best_post= 98.67 | best_any= 98.67\n",
      "Client   1 | best_pre= 97.93 | best_post= 97.30 | best_any= 97.93\n",
      "Client   2 | best_pre= 98.40 | best_post= 96.87 | best_any= 98.40\n",
      "Client   3 | best_pre= 84.23 | best_post= 80.33 | best_any= 84.23\n",
      "Client   4 | best_pre= 95.07 | best_post= 93.93 | best_any= 95.07\n",
      "Client   5 | best_pre= 82.33 | best_post= 81.90 | best_any= 82.33\n",
      "Client   6 | best_pre= 83.97 | best_post= 83.07 | best_any= 83.97\n",
      "Client   7 | best_pre= 98.40 | best_post= 98.33 | best_any= 98.40\n",
      "Client   8 | best_pre= 98.47 | best_post= 98.23 | best_any= 98.47\n",
      "Client   9 | best_pre= 94.77 | best_post= 93.43 | best_any= 94.77\n",
      "Client  10 | best_pre= 98.43 | best_post= 98.33 | best_any= 98.43\n",
      "Client  11 | best_pre= 98.53 | best_post= 98.13 | best_any= 98.53\n",
      "Client  12 | best_pre= 84.23 | best_post= 79.23 | best_any= 84.23\n",
      "Client  13 | best_pre= 83.37 | best_post= 70.63 | best_any= 83.37\n",
      "Client  14 | best_pre= 94.67 | best_post= 90.93 | best_any= 94.67\n",
      "Client  15 | best_pre= 83.97 | best_post= 81.60 | best_any= 83.97\n",
      "Client  16 | best_pre= 95.07 | best_post= 95.43 | best_any= 95.43\n",
      "Client  17 | best_pre= 98.53 | best_post= 90.90 | best_any= 98.53\n",
      "Client  18 | best_pre= 82.67 | best_post= 74.20 | best_any= 82.67\n",
      "Client  19 | best_pre= 98.80 | best_post= 98.07 | best_any= 98.80\n",
      "Client  20 | best_pre= 98.20 | best_post= 98.37 | best_any= 98.37\n",
      "Client  21 | best_pre= 98.80 | best_post= 98.57 | best_any= 98.80\n",
      "Client  22 | best_pre= 92.40 | best_post= 90.20 | best_any= 92.40\n",
      "Client  23 | best_pre= 98.70 | best_post= 98.10 | best_any= 98.70\n",
      "Client  24 | best_pre= 98.53 | best_post= 98.60 | best_any= 98.60\n",
      "Client  25 | best_pre= 98.67 | best_post= 98.50 | best_any= 98.67\n",
      "Client  26 | best_pre= 94.93 | best_post= 95.33 | best_any= 95.33\n",
      "Client  27 | best_pre= 98.57 | best_post= 97.93 | best_any= 98.57\n",
      "Client  28 | best_pre= 83.87 | best_post= 81.33 | best_any= 83.87\n",
      "Client  29 | best_pre= 98.27 | best_post= 94.43 | best_any= 98.27\n",
      "Client  30 | best_pre= 98.57 | best_post= 98.63 | best_any= 98.63\n",
      "Client  31 | best_pre= 94.93 | best_post= 93.87 | best_any= 94.93\n",
      "Client  32 | best_pre= 82.67 | best_post= 90.03 | best_any= 90.03\n",
      "Client  33 | best_pre= 98.67 | best_post= 97.50 | best_any= 98.67\n",
      "Client  34 | best_pre= 98.13 | best_post= 97.27 | best_any= 98.13\n",
      "Client  35 | best_pre= 84.23 | best_post= 81.27 | best_any= 84.23\n",
      "Client  36 | best_pre= 98.53 | best_post= 98.70 | best_any= 98.70\n",
      "Client  37 | best_pre= 98.37 | best_post= 98.53 | best_any= 98.53\n",
      "Client  38 | best_pre= 98.70 | best_post= 98.60 | best_any= 98.70\n",
      "Client  39 | best_pre= 98.67 | best_post= 98.67 | best_any= 98.67\n",
      "Client  40 | best_pre= 86.23 | best_post= 89.80 | best_any= 89.80\n",
      "Client  41 | best_pre= 98.60 | best_post= 98.33 | best_any= 98.60\n",
      "Client  42 | best_pre= 98.47 | best_post= 98.47 | best_any= 98.47\n",
      "Client  43 | best_pre= 94.67 | best_post= 94.73 | best_any= 94.73\n",
      "Client  44 | best_pre= 87.80 | best_post= 91.33 | best_any= 91.33\n",
      "Client  45 | best_pre= 98.40 | best_post= 98.47 | best_any= 98.47\n",
      "Client  46 | best_pre= 94.67 | best_post= 95.00 | best_any= 95.00\n",
      "Client  47 | best_pre= 98.57 | best_post= 98.20 | best_any= 98.57\n",
      "Client  48 | best_pre= 94.20 | best_post= 92.20 | best_any= 94.20\n",
      "Client  49 | best_pre= 94.07 | best_post= 93.60 | best_any= 94.07\n",
      "Client  50 | best_pre= 81.63 | best_post= 90.93 | best_any= 90.93\n",
      "Client  51 | best_pre= 98.57 | best_post= 98.03 | best_any= 98.57\n",
      "Client  52 | best_pre= 82.37 | best_post= 75.13 | best_any= 82.37\n",
      "Client  53 | best_pre= 83.87 | best_post= 71.30 | best_any= 83.87\n",
      "Client  54 | best_pre= 98.53 | best_post= 98.27 | best_any= 98.53\n",
      "Client  55 | best_pre= 83.67 | best_post= 76.57 | best_any= 83.67\n",
      "Client  56 | best_pre= 94.67 | best_post= 92.77 | best_any= 94.67\n",
      "Client  57 | best_pre= 98.53 | best_post= 98.37 | best_any= 98.53\n",
      "Client  58 | best_pre= 98.57 | best_post= 97.00 | best_any= 98.57\n",
      "Client  59 | best_pre= 71.63 | best_post= 87.70 | best_any= 87.70\n",
      "Client  60 | best_pre= 98.27 | best_post= 98.40 | best_any= 98.40\n",
      "Client  61 | best_pre= 82.17 | best_post= 88.23 | best_any= 88.23\n",
      "Client  62 | best_pre= 98.47 | best_post= 98.40 | best_any= 98.47\n",
      "Client  63 | best_pre= 98.60 | best_post= 96.90 | best_any= 98.60\n",
      "Client  64 | best_pre= 92.40 | best_post= 89.50 | best_any= 92.40\n",
      "Client  65 | best_pre= 86.23 | best_post= 87.53 | best_any= 87.53\n",
      "Client  66 | best_pre= 83.97 | best_post= 61.53 | best_any= 83.97\n",
      "Client  67 | best_pre= 98.67 | best_post= 91.43 | best_any= 98.67\n",
      "Client  68 | best_pre= 79.63 | best_post= 88.20 | best_any= 88.20\n",
      "Client  69 | best_pre= 98.60 | best_post= 98.47 | best_any= 98.60\n",
      "Client  70 | best_pre= 98.80 | best_post= 98.67 | best_any= 98.80\n",
      "Client  71 | best_pre= 83.97 | best_post= 78.63 | best_any= 83.97\n",
      "Client  72 | best_pre= 84.23 | best_post= 82.00 | best_any= 84.23\n",
      "Client  73 | best_pre= 83.97 | best_post= 72.07 | best_any= 83.97\n",
      "Client  74 | best_pre= 98.47 | best_post= 95.03 | best_any= 98.47\n",
      "Client  75 | best_pre= 98.67 | best_post= 98.60 | best_any= 98.67\n",
      "Client  76 | best_pre= 95.07 | best_post= 94.30 | best_any= 95.07\n",
      "Client  77 | best_pre= 98.80 | best_post= 98.37 | best_any= 98.80\n",
      "Client  78 | best_pre= 98.60 | best_post= 97.97 | best_any= 98.60\n",
      "Client  79 | best_pre= 86.23 | best_post= 90.27 | best_any= 90.27\n",
      "Client  80 | best_pre= 84.23 | best_post= 80.53 | best_any= 84.23\n",
      "Client  81 | best_pre= 83.87 | best_post= 79.73 | best_any= 83.87\n",
      "Client  82 | best_pre= 74.77 | best_post= 89.73 | best_any= 89.73\n",
      "Client  83 | best_pre= 83.73 | best_post= 81.23 | best_any= 83.73\n",
      "Client  84 | best_pre= 98.50 | best_post= 97.73 | best_any= 98.50\n",
      "Client  85 | best_pre= 98.70 | best_post= 98.57 | best_any= 98.70\n",
      "Client  86 | best_pre= 98.80 | best_post= 98.47 | best_any= 98.80\n",
      "Client  87 | best_pre= 94.23 | best_post= 93.33 | best_any= 94.23\n",
      "Client  88 | best_pre= 95.07 | best_post= 95.13 | best_any= 95.13\n",
      "Client  89 | best_pre= 98.80 | best_post= 98.17 | best_any= 98.80\n",
      "Client  90 | best_pre= 83.07 | best_post= 91.27 | best_any= 91.27\n",
      "Client  91 | best_pre= 95.07 | best_post= 90.83 | best_any= 95.07\n",
      "Client  92 | best_pre= 98.63 | best_post= 98.07 | best_any= 98.63\n",
      "Client  93 | best_pre= 98.13 | best_post= 91.37 | best_any= 98.13\n",
      "Client  94 | best_pre= 81.63 | best_post= 90.90 | best_any= 90.90\n",
      "Client  95 | best_pre= 98.67 | best_post= 97.83 | best_any= 98.67\n",
      "Client  96 | best_pre= 98.47 | best_post= 98.47 | best_any= 98.47\n",
      "Client  97 | best_pre= 83.67 | best_post= 75.77 | best_any= 83.67\n",
      "Client  98 | best_pre= 87.80 | best_post= 86.57 | best_any= 87.80\n",
      "Client  99 | best_pre= 98.70 | best_post= 98.37 | best_any= 98.70\n",
      "\n",
      "##### ROUND 32 #####\n",
      "\n",
      "##### ROUND 33 #####\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[8], line 97\u001b[0m\n\u001b[1;32m     95\u001b[0m sec_updates, data_sizes \u001b[38;5;241m=\u001b[39m [], []\n\u001b[1;32m     96\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m uid \u001b[38;5;129;01min\u001b[39;00m clist:\n\u001b[0;32m---> 97\u001b[0m     sd_sec \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[43mtrain_secondary\u001b[49m\u001b[43m(\u001b[49m\n\u001b[1;32m     98\u001b[0m \u001b[43m        \u001b[49m\u001b[43mnew_sec_encoder\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43msec_template\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m     99\u001b[0m \u001b[43m        \u001b[49m\u001b[43mepochs\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mlocal_sec_ep\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    100\u001b[0m \u001b[43m        \u001b[49m\u001b[43mlr\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43msec_lr\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    101\u001b[0m \u001b[43m        \u001b[49m\u001b[43mreturn_state_dict\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mTrue\u001b[39;49;00m\u001b[43m)\u001b[49m\n\u001b[1;32m    102\u001b[0m     sec_updates\u001b[38;5;241m.\u001b[39mappend(sd_sec)\n\u001b[1;32m    103\u001b[0m     data_sizes\u001b[38;5;241m.\u001b[39mappend(\u001b[38;5;28mlen\u001b[39m(net_dataidx_map[uid]))\n",
      "File \u001b[0;32m~/Misc/PACFLComboNB/Flag/src/client/client_cluster_fl.py:132\u001b[0m, in \u001b[0;36mClient_ClusterFL.train_secondary\u001b[0;34m(self, new_sec_encoder, epochs, lr, return_state_dict)\u001b[0m\n\u001b[1;32m    130\u001b[0m         opt\u001b[38;5;241m.\u001b[39mzero_grad()\n\u001b[1;32m    131\u001b[0m         loss \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mloss_func(temp_model(x), y)\n\u001b[0;32m--> 132\u001b[0m         \u001b[43mloss\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mbackward\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    133\u001b[0m         opt\u001b[38;5;241m.\u001b[39mstep()\n\u001b[1;32m    135\u001b[0m \u001b[38;5;66;03m# 5. Return enriched secondary encoder\u001b[39;00m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/_tensor.py:525\u001b[0m, in \u001b[0;36mTensor.backward\u001b[0;34m(self, gradient, retain_graph, create_graph, inputs)\u001b[0m\n\u001b[1;32m    515\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m has_torch_function_unary(\u001b[38;5;28mself\u001b[39m):\n\u001b[1;32m    516\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m handle_torch_function(\n\u001b[1;32m    517\u001b[0m         Tensor\u001b[38;5;241m.\u001b[39mbackward,\n\u001b[1;32m    518\u001b[0m         (\u001b[38;5;28mself\u001b[39m,),\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m    523\u001b[0m         inputs\u001b[38;5;241m=\u001b[39minputs,\n\u001b[1;32m    524\u001b[0m     )\n\u001b[0;32m--> 525\u001b[0m \u001b[43mtorch\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mautograd\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mbackward\u001b[49m\u001b[43m(\u001b[49m\n\u001b[1;32m    526\u001b[0m \u001b[43m    \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mgradient\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mretain_graph\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mcreate_graph\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43minputs\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43minputs\u001b[49m\n\u001b[1;32m    527\u001b[0m \u001b[43m\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/autograd/__init__.py:267\u001b[0m, in \u001b[0;36mbackward\u001b[0;34m(tensors, grad_tensors, retain_graph, create_graph, grad_variables, inputs)\u001b[0m\n\u001b[1;32m    262\u001b[0m     retain_graph \u001b[38;5;241m=\u001b[39m create_graph\n\u001b[1;32m    264\u001b[0m \u001b[38;5;66;03m# The reason we repeat the same comment below is that\u001b[39;00m\n\u001b[1;32m    265\u001b[0m \u001b[38;5;66;03m# some Python versions print out the first line of a multi-line function\u001b[39;00m\n\u001b[1;32m    266\u001b[0m \u001b[38;5;66;03m# calls in the traceback and some print out the last line\u001b[39;00m\n\u001b[0;32m--> 267\u001b[0m \u001b[43m_engine_run_backward\u001b[49m\u001b[43m(\u001b[49m\n\u001b[1;32m    268\u001b[0m \u001b[43m    \u001b[49m\u001b[43mtensors\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    269\u001b[0m \u001b[43m    \u001b[49m\u001b[43mgrad_tensors_\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    270\u001b[0m \u001b[43m    \u001b[49m\u001b[43mretain_graph\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    271\u001b[0m \u001b[43m    \u001b[49m\u001b[43mcreate_graph\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    272\u001b[0m \u001b[43m    \u001b[49m\u001b[43minputs\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m    273\u001b[0m \u001b[43m    \u001b[49m\u001b[43mallow_unreachable\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mTrue\u001b[39;49;00m\u001b[43m,\u001b[49m\n\u001b[1;32m    274\u001b[0m \u001b[43m    \u001b[49m\u001b[43maccumulate_grad\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mTrue\u001b[39;49;00m\u001b[43m,\u001b[49m\n\u001b[1;32m    275\u001b[0m \u001b[43m\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/autograd/graph.py:744\u001b[0m, in \u001b[0;36m_engine_run_backward\u001b[0;34m(t_outputs, *args, **kwargs)\u001b[0m\n\u001b[1;32m    742\u001b[0m     unregister_hooks \u001b[38;5;241m=\u001b[39m _register_logging_hooks_on_whole_graph(t_outputs)\n\u001b[1;32m    743\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[0;32m--> 744\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mVariable\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_execution_engine\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mrun_backward\u001b[49m\u001b[43m(\u001b[49m\u001b[43m  \u001b[49m\u001b[38;5;66;43;03m# Calls into the C++ engine to run the backward pass\u001b[39;49;00m\n\u001b[1;32m    745\u001b[0m \u001b[43m        \u001b[49m\u001b[43mt_outputs\u001b[49m\u001b[43m,\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\n\u001b[1;32m    746\u001b[0m \u001b[43m    \u001b[49m\u001b[43m)\u001b[49m  \u001b[38;5;66;03m# Calls into the C++ engine to run the backward pass\u001b[39;00m\n\u001b[1;32m    747\u001b[0m \u001b[38;5;28;01mfinally\u001b[39;00m:\n\u001b[1;32m    748\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m attach_logging_hooks:\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "# ============================================================\n",
    "# Clustered-FL primary-update loop\n",
    "# ============================================================\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():           # ← <-- changed\n",
    "        # z   : cluster-id that has at least one sampled client this round\n",
    "        # clist_sampled : the *subset* of clients from that cluster that\n",
    "        #                 appeared in idxs_users\n",
    "    \n",
    "        learners = H_out.get(z, [])                         # clusters j with H(j,z)=1\n",
    "        if not learners or len(clist_sampled) == 0:\n",
    "            continue                                        # nothing to do\n",
    "    \n",
    "        # ---------- 6.1  Θ2′z : average of current learner-clusters ----------\n",
    "        learner_secs = [\n",
    "            OrderedDict({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",
    "        # ---------- 6.2  build a template encoder with Θ2′z ----------\n",
    "        sec_template = clone_encoder(\n",
    "            clients[clist_sampled[0]].net.secondary_encoder)\n",
    "        stripped = OrderedDict(\n",
    "            (k.replace(\"secondary_encoder.\", \"\"), v) for k, v in Θ2_prime.items()\n",
    "        )\n",
    "        sec_template.load_state_dict(stripped)\n",
    "    \n",
    "        # ---------- 6.3  local enrichment ONLY on sampled clients ----------\n",
    "        sec_updates, data_sizes = [], []\n",
    "        for uid in clist_sampled:                           # ← <-- changed\n",
    "            sd_sec = clients[uid].train_secondary(\n",
    "                new_sec_encoder=sec_template,\n",
    "                epochs=local_sec_ep,\n",
    "                lr=sec_lr,\n",
    "                return_state_dict=True)\n",
    "            sec_updates.append(sd_sec)\n",
    "            data_sizes.append(len(net_dataidx_map[uid]))\n",
    "    \n",
    "        # ---------- 6.4  FedAvg over those updates ----------\n",
    "        theta_bar = fedavg_secondary(sec_updates, data_sizes)\n",
    "    \n",
    "        # ---------- 6.5  push θ̄2′(z) to every learner cluster j ----------\n",
    "        for j in learners:\n",
    "            for k, v in theta_bar.items():                  # k = \"0.weight\", …\n",
    "                key = f\"secondary_encoder.{k}\"              # add prefix\n",
    "                w_glob_per_cluster[j][key] = v.clone()\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
}
