{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 9,
   "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='fmnist', 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='fmnist'\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": 10,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "partition: flag-non-iid\n",
      "Data statistics Train:\n",
      " {0: {4: 2, 8: 115, 9: 119}, 1: {4: 128, 8: 5, 9: 442}, 2: {0: 79, 1: 126, 3: 89}, 3: {2: 1, 5: 475, 7: 397}, 4: {4: 91, 8: 119, 9: 345}, 5: {2: 118, 5: 387, 7: 210}, 6: {0: 115, 1: 78, 3: 43}, 7: {4: 50, 8: 158, 9: 1}, 8: {1: 350, 4: 88, 7: 62}, 9: {4: 153, 8: 57, 9: 16}, 10: {0: 13, 1: 24, 3: 16}, 11: {2: 426, 5: 37, 7: 55}, 12: {2: 8, 5: 777, 7: 67}, 13: {0: 66, 1: 64, 3: 277}, 14: {0: 54, 4: 149, 6: 188}, 15: {0: 54, 4: 69, 6: 40}, 16: {0: 303, 1: 225, 3: 36}, 17: {4: 22, 8: 32, 9: 517}, 18: {0: 79, 1: 66, 3: 36}, 19: {4: 2, 8: 67, 9: 74}, 20: {1: 49, 4: 54, 7: 103}, 21: {0: 85, 4: 14, 6: 339}, 22: {4: 92, 8: 175, 9: 666}, 23: {0: 27, 1: 155, 3: 613}, 24: {0: 80, 1: 481, 3: 231}, 25: {0: 124, 1: 36, 3: 75}, 26: {0: 284, 4: 39, 6: 36}, 27: {2: 660, 5: 15, 7: 278}, 28: {0: 24, 1: 254, 3: 652}, 29: {4: 80, 8: 167, 9: 152}, 30: {1: 5, 4: 12, 7: 229}, 31: {1: 1, 4: 81, 7: 4}, 32: {0: 376, 1: 202, 3: 9}, 33: {0: 113, 1: 523, 3: 590}, 34: {0: 111, 4: 103, 6: 988}, 35: {0: 197, 4: 33, 6: 1}, 36: {2: 849, 5: 110, 7: 127}, 37: {4: 390, 8: 331, 9: 5}, 38: {0: 28, 1: 158, 3: 13}, 39: {0: 65, 4: 136, 6: 62}, 40: {1: 300, 4: 317, 7: 152}, 41: {4: 133, 8: 1211, 9: 51}, 42: {0: 126, 4: 267, 6: 13}, 43: {4: 149, 8: 701, 9: 597}, 44: {0: 16, 1: 272, 3: 141}, 45: {2: 10, 5: 341, 7: 27}, 46: {2: 768, 5: 419, 7: 488}, 47: {0: 103, 4: 103, 6: 282}, 48: {0: 46, 1: 101, 3: 107}, 49: {1: 32, 4: 79, 7: 42}, 50: {0: 52, 1: 63, 3: 1242}, 51: {1: 136, 4: 60, 7: 39}, 52: {1: 83, 4: 151, 7: 416}, 53: {4: 166, 8: 589, 9: 74}, 54: {1: 81, 4: 22, 7: 143}, 55: {1: 172, 4: 140, 7: 156}, 56: {0: 531, 1: 289, 3: 128}, 57: {0: 43, 4: 21, 6: 350}, 58: {0: 244, 4: 15, 6: 17}, 59: {4: 58, 8: 31, 9: 786}, 60: {1: 234, 4: 239, 7: 190}, 61: {0: 291, 4: 285, 6: 257}, 62: {2: 335, 5: 34, 7: 95}, 63: {4: 97, 8: 102, 9: 385}, 64: {0: 66, 1: 6, 3: 439}, 65: {4: 29, 8: 8, 9: 44}, 66: {1: 165, 4: 304, 7: 17}, 67: {0: 11, 4: 293, 6: 61}, 68: {2: 226, 5: 15, 7: 393}, 69: {4: 176, 8: 43, 9: 332}, 70: {1: 245, 4: 28, 7: 99}, 71: {1: 451, 4: 99, 7: 4}, 72: {2: 244, 5: 246, 7: 40}, 73: {4: 406, 8: 224, 9: 239}, 74: {0: 3, 4: 11, 6: 1332}, 75: {0: 120, 4: 54, 6: 380}, 76: {0: 131, 1: 35, 3: 71}, 77: {0: 11, 4: 16, 6: 482}, 78: {0: 15, 4: 26, 6: 863}, 79: {2: 216, 5: 300, 7: 472}, 80: {2: 1139, 5: 171, 7: 100}, 81: {0: 304, 1: 58, 3: 100}, 82: {1: 161, 4: 77, 7: 41}, 83: {4: 20, 8: 621, 9: 191}, 84: {2: 165, 5: 169, 7: 181}, 85: {2: 507, 5: 666, 7: 467}, 86: {4: 5, 8: 644, 9: 342}, 87: {4: 157, 8: 381, 9: 419}, 88: {0: 93, 1: 3, 3: 347}, 89: {0: 744, 4: 103, 6: 35}, 90: {0: 351, 4: 67, 6: 256}, 91: {0: 211, 1: 142, 3: 336}, 92: {0: 46, 1: 120, 3: 205}, 93: {0: 92, 1: 54, 3: 204}, 94: {2: 114, 5: 374, 7: 94}, 95: {2: 150, 5: 903, 7: 136}, 96: {0: 73, 4: 14, 6: 18}, 97: {4: 25, 8: 219, 9: 203}, 98: {2: 19, 5: 94, 7: 376}, 99: {2: 45, 5: 467, 7: 300}} \n",
      "\n",
      "Data statistics Test:\n",
      " {0: {4: 1000, 8: 1000, 9: 1000}, 1: {4: 1000, 8: 1000, 9: 1000}, 2: {0: 1000, 1: 1000, 3: 1000}, 3: {2: 1000, 5: 1000, 7: 1000}, 4: {4: 1000, 8: 1000, 9: 1000}, 5: {2: 1000, 5: 1000, 7: 1000}, 6: {0: 1000, 1: 1000, 3: 1000}, 7: {4: 1000, 8: 1000, 9: 1000}, 8: {1: 1000, 4: 1000, 7: 1000}, 9: {4: 1000, 8: 1000, 9: 1000}, 10: {0: 1000, 1: 1000, 3: 1000}, 11: {2: 1000, 5: 1000, 7: 1000}, 12: {2: 1000, 5: 1000, 7: 1000}, 13: {0: 1000, 1: 1000, 3: 1000}, 14: {0: 1000, 4: 1000, 6: 1000}, 15: {0: 1000, 4: 1000, 6: 1000}, 16: {0: 1000, 1: 1000, 3: 1000}, 17: {4: 1000, 8: 1000, 9: 1000}, 18: {0: 1000, 1: 1000, 3: 1000}, 19: {4: 1000, 8: 1000, 9: 1000}, 20: {1: 1000, 4: 1000, 7: 1000}, 21: {0: 1000, 4: 1000, 6: 1000}, 22: {4: 1000, 8: 1000, 9: 1000}, 23: {0: 1000, 1: 1000, 3: 1000}, 24: {0: 1000, 1: 1000, 3: 1000}, 25: {0: 1000, 1: 1000, 3: 1000}, 26: {0: 1000, 4: 1000, 6: 1000}, 27: {2: 1000, 5: 1000, 7: 1000}, 28: {0: 1000, 1: 1000, 3: 1000}, 29: {4: 1000, 8: 1000, 9: 1000}, 30: {1: 1000, 4: 1000, 7: 1000}, 31: {1: 1000, 4: 1000, 7: 1000}, 32: {0: 1000, 1: 1000, 3: 1000}, 33: {0: 1000, 1: 1000, 3: 1000}, 34: {0: 1000, 4: 1000, 6: 1000}, 35: {0: 1000, 4: 1000, 6: 1000}, 36: {2: 1000, 5: 1000, 7: 1000}, 37: {4: 1000, 8: 1000, 9: 1000}, 38: {0: 1000, 1: 1000, 3: 1000}, 39: {0: 1000, 4: 1000, 6: 1000}, 40: {1: 1000, 4: 1000, 7: 1000}, 41: {4: 1000, 8: 1000, 9: 1000}, 42: {0: 1000, 4: 1000, 6: 1000}, 43: {4: 1000, 8: 1000, 9: 1000}, 44: {0: 1000, 1: 1000, 3: 1000}, 45: {2: 1000, 5: 1000, 7: 1000}, 46: {2: 1000, 5: 1000, 7: 1000}, 47: {0: 1000, 4: 1000, 6: 1000}, 48: {0: 1000, 1: 1000, 3: 1000}, 49: {1: 1000, 4: 1000, 7: 1000}, 50: {0: 1000, 1: 1000, 3: 1000}, 51: {1: 1000, 4: 1000, 7: 1000}, 52: {1: 1000, 4: 1000, 7: 1000}, 53: {4: 1000, 8: 1000, 9: 1000}, 54: {1: 1000, 4: 1000, 7: 1000}, 55: {1: 1000, 4: 1000, 7: 1000}, 56: {0: 1000, 1: 1000, 3: 1000}, 57: {0: 1000, 4: 1000, 6: 1000}, 58: {0: 1000, 4: 1000, 6: 1000}, 59: {4: 1000, 8: 1000, 9: 1000}, 60: {1: 1000, 4: 1000, 7: 1000}, 61: {0: 1000, 4: 1000, 6: 1000}, 62: {2: 1000, 5: 1000, 7: 1000}, 63: {4: 1000, 8: 1000, 9: 1000}, 64: {0: 1000, 1: 1000, 3: 1000}, 65: {4: 1000, 8: 1000, 9: 1000}, 66: {1: 1000, 4: 1000, 7: 1000}, 67: {0: 1000, 4: 1000, 6: 1000}, 68: {2: 1000, 5: 1000, 7: 1000}, 69: {4: 1000, 8: 1000, 9: 1000}, 70: {1: 1000, 4: 1000, 7: 1000}, 71: {1: 1000, 4: 1000, 7: 1000}, 72: {2: 1000, 5: 1000, 7: 1000}, 73: {4: 1000, 8: 1000, 9: 1000}, 74: {0: 1000, 4: 1000, 6: 1000}, 75: {0: 1000, 4: 1000, 6: 1000}, 76: {0: 1000, 1: 1000, 3: 1000}, 77: {0: 1000, 4: 1000, 6: 1000}, 78: {0: 1000, 4: 1000, 6: 1000}, 79: {2: 1000, 5: 1000, 7: 1000}, 80: {2: 1000, 5: 1000, 7: 1000}, 81: {0: 1000, 1: 1000, 3: 1000}, 82: {1: 1000, 4: 1000, 7: 1000}, 83: {4: 1000, 8: 1000, 9: 1000}, 84: {2: 1000, 5: 1000, 7: 1000}, 85: {2: 1000, 5: 1000, 7: 1000}, 86: {4: 1000, 8: 1000, 9: 1000}, 87: {4: 1000, 8: 1000, 9: 1000}, 88: {0: 1000, 1: 1000, 3: 1000}, 89: {0: 1000, 4: 1000, 6: 1000}, 90: {0: 1000, 4: 1000, 6: 1000}, 91: {0: 1000, 1: 1000, 3: 1000}, 92: {0: 1000, 1: 1000, 3: 1000}, 93: {0: 1000, 1: 1000, 3: 1000}, 94: {2: 1000, 5: 1000, 7: 1000}, 95: {2: 1000, 5: 1000, 7: 1000}, 96: {0: 1000, 4: 1000, 6: 1000}, 97: {4: 1000, 8: 1000, 9: 1000}, 98: {2: 1000, 5: 1000, 7: 1000}, 99: {2: 1000, 5: 1000, 7: 1000}} \n",
      "\n",
      "len train_ds_global: 60000\n",
      "len test_ds_global: 10000\n",
      "MODEL: simple-cnn, Dataset: fmnist\n",
      "SimpleCNNMNIST2(\n",
      "  (encoder): Sequential(\n",
      "    (0): Conv2d(1, 6, kernel_size=(5, 5), stride=(1, 1), padding=(2, 2))\n",
      "    (1): ReLU()\n",
      "    (2): MaxPool2d(kernel_size=2, stride=2, padding=0, dilation=1, ceil_mode=False)\n",
      "    (3): Conv2d(6, 16, kernel_size=(5, 5), stride=(1, 1))\n",
      "    (4): ReLU()\n",
      "    (5): MaxPool2d(kernel_size=2, stride=2, padding=0, dilation=1, ceil_mode=False)\n",
      "    (6): Flatten(start_dim=1, end_dim=-1)\n",
      "    (7): Linear(in_features=400, out_features=120, bias=True)\n",
      "    (8): ReLU()\n",
      "    (9): Linear(in_features=120, out_features=84, bias=True)\n",
      "    (10): ReLU()\n",
      "  )\n",
      "  (classifier): Linear(in_features=84, out_features=10, bias=True)\n",
      ")\n",
      "encoder.0.weight torch.Size([6, 1, 5, 5])\n",
      "encoder.0.bias torch.Size([6])\n",
      "encoder.3.weight torch.Size([16, 6, 5, 5])\n",
      "encoder.3.bias torch.Size([16])\n",
      "encoder.7.weight torch.Size([120, 400])\n",
      "encoder.7.bias torch.Size([120])\n",
      "encoder.9.weight torch.Size([84, 120])\n",
      "encoder.9.bias torch.Size([84])\n",
      "classifier.weight torch.Size([10, 84])\n",
      "classifier.bias torch.Size([10])\n",
      "61706\n",
      "Labels: [4 8 9], Counts: [  2 115 119]\n",
      "Shape of U: (784, 8)\n",
      "Labels: [4 8 9], Counts: [128   5 442]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [ 79 126  89]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [  1 475 397]\n",
      "Shape of U: (784, 7)\n",
      "Labels: [4 8 9], Counts: [ 91 119 345]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [118 387 210]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [115  78  43]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [ 50 158   1]\n",
      "Shape of U: (784, 7)\n",
      "Labels: [1 4 7], Counts: [350  88  62]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [153  57  16]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [13 24 16]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [426  37  55]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [  8 777  67]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [ 66  64 277]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [ 54 149 188]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [54 69 40]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [303 225  36]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [ 22  32 517]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [79 66 36]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [ 2 67 74]\n",
      "Shape of U: (784, 8)\n",
      "Labels: [1 4 7], Counts: [ 49  54 103]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [ 85  14 339]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [ 92 175 666]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [ 27 155 613]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [ 80 481 231]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [124  36  75]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [284  39  36]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [660  15 278]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [ 24 254 652]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [ 80 167 152]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [1 4 7], Counts: [  5  12 229]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [1 4 7], Counts: [ 1 81  4]\n",
      "Shape of U: (784, 7)\n",
      "Labels: [0 1 3], Counts: [376 202   9]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [113 523 590]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [111 103 988]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [197  33   1]\n",
      "Shape of U: (784, 7)\n",
      "Labels: [2 5 7], Counts: [849 110 127]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [390 331   5]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [ 28 158  13]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [ 65 136  62]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [1 4 7], Counts: [300 317 152]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [ 133 1211   51]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [126 267  13]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [149 701 597]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [ 16 272 141]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [ 10 341  27]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [768 419 488]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [103 103 282]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [ 46 101 107]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [1 4 7], Counts: [32 79 42]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [  52   63 1242]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [1 4 7], Counts: [136  60  39]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [1 4 7], Counts: [ 83 151 416]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [166 589  74]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [1 4 7], Counts: [ 81  22 143]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [1 4 7], Counts: [172 140 156]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [531 289 128]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [ 43  21 350]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [244  15  17]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [ 58  31 786]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [1 4 7], Counts: [234 239 190]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [291 285 257]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [335  34  95]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [ 97 102 385]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [ 66   6 439]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [29  8 44]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [1 4 7], Counts: [165 304  17]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [ 11 293  61]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [226  15 393]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [176  43 332]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [1 4 7], Counts: [245  28  99]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [1 4 7], Counts: [451  99   4]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [244 246  40]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [406 224 239]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [   3   11 1332]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [120  54 380]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [131  35  71]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [ 11  16 482]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [ 15  26 863]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [216 300 472]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [1139  171  100]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [304  58 100]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [1 4 7], Counts: [161  77  41]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [ 20 621 191]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [165 169 181]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [507 666 467]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [  5 644 342]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [157 381 419]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [ 93   3 347]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [744 103  35]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [351  67 256]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [211 142 336]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [ 46 120 205]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 3], Counts: [ 92  54 204]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [114 374  94]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [150 903 136]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 6], Counts: [73 14 18]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [4 8 9], Counts: [ 25 219 203]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [ 19  94 376]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 5 7], Counts: [ 45 467 300]\n",
      "Shape of U: (784, 9)\n",
      "###### Gradient data ROUND 1 ######\n",
      "Grad_similarities:\n",
      "[[ 0.0000 52.3706 72.9953 ... 28.3817 76.6300 88.6742]\n",
      " [52.3706  0.0000 73.3495 ... 50.8193 75.5919 84.0121]\n",
      " [72.9953 73.3495  0.0000 ... 76.4212 75.3180 84.2171]\n",
      " ...\n",
      " [28.3817 50.8193 76.4212 ...  0.0000 79.2230 87.6751]\n",
      " [76.6300 75.5919 75.3180 ... 79.2230  0.0000 45.0904]\n",
      " [88.6742 84.0121 84.2171 ... 87.6751 45.0904  0.0000]]\n"
     ]
    }
   ],
   "source": [
    "##################################### Computing data and gradient matrix\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)\n",
    "\n",
    "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",
    "compression_ratio = 50           # 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(2): \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": 11,
   "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 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.9695\n",
      " 0.0000 0.0000 0.0000 0.0000 1.0000 0.0000 0.0000 0.0000 0.0000 0.0000\n",
      " 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 1.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",
      " 1.0000 0.0000 0.0000 0.0000 0.0000 0.0000 0.0000 1.0000 1.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"
     ]
    }
   ],
   "source": [
    "#Fusion-gate: combing data and gradient matrix\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": 12,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO:matplotlib.mathtext:Substituting symbol L from STIXNonUnicode\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n",
      "INFO:matplotlib.mathtext:Substituting symbol L from STIXNonUnicode\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n",
      "INFO:matplotlib.mathtext:Substituting symbol C from STIXGeneral\n",
      "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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WuD5S5cuqYSEBAVAPfvss0W25+2vYWpqqvX/yoQJE9x+vTf616xZM4VCyhtduXJF1atXTwFQvXv39nn/CpOWlqaaNGmiAKgBAwa4/Xojkqwa+TlCxmHwZZAtW7Y4rRn20EMPKUBqj+Uvd/P++++rGjVqOPxloJRSbdu2VSiiJpcrdQm91b/89ARf3ujjX3/9ZS2a27dvX48DL2/175lnnlGLFy9WGRkZeR7PzMxU3333nbXAdrt27YosTeKr77HG3eDLG/17/vnn1TfffKNSUlLyPH7q1Cn12GOPWT+4PvjgA1P6p5StPE6ZMmXyBNAXLlywBoclSpRQ+/fvN62PmilTpihAii+fOnWqyP74on+TJk2yfh9feOGFPN/rU6dOWQMqQMpe+bp/Sin13nvvqRMnTuR5bPv27apTp07WjPmuFne352rw5avPETIOgy8Dffnllyo0NFQBUoesZcuW1g/+6Oho9euvvxZ4jVa+o0aNGg7bPHTokKpevbr1F2L+avRPPPGEqf274447VNmyZa03rcZayZIl8zxu/x/el31MTEy0fq3atm2rOnTo4PRmRv9q1Khh/d42aNBAtW3bVjVr1sxaUBuAat26tTp58qQp/SuMu8GXN/qn9SE0NFTVq1dPXXfddaphw4bWa1gsFjVs2DDT+qeUBNK333679ftZr1491apVK2upm5iYGJcLa3urj5qOHTsqAOruu+92uT/e7l92dra67777rF+/qKgo1ahRI5WYmGgNHADXa2d64+un/axVqVJFtW7dWtWuXdvar4YNG6qDBw+61LfDhw/n+b0ZHx9vbcf+8TvuuMOt/hn5OULGYPBlsL/++kvdc889qkKFCioiIkJVqVJF9evXT+3atcvh+a78Urxw4YIaNmyYSkxMVFFRUapUqVKqc+fOasaMGab3T/vwK+rmTt09I/uoBTeu3Mzo34IFC9Tzzz+vWrVqpSpVqqTCw8NVdHS0qlmzprr77rvVjBkz3J4K8MbPoCOeBF9G92/+/PlqwIAB6tprr1UVK1ZUERERKiYmRiUmJqrHH39crV+/3q2+Gd0/TW5urpoyZYq6/vrrValSpVRERISqWbOmevLJJ9WePXv8oo/2NQgXLVrkdp+83b958+apXr16qSpVqqiIiAgVGRmpatasqfr27atWrlxpav+SkpJU586dVcWKFVV4eLgqW7as6tSpk/rwww+LnO62d+DAAZd+V3Xu3Nmt/ill7OcI6WdRysmeWyIiIiIyHHc7EhEREfkQgy8iIiIiH2LwRURERORDDL6IiIiIfIjBFxEREZEPMfgiIiIi8iEGX0REREQ+xOCLiIiIyIcYfBERERH5EIMvIiIiIh9i8EV+ZerUqbBYLAVu0dHRqFixIho1aoQHHngA7733Ho4cOVJke8uXL7e2cfDgQe+/AfKK4vB99If3ePDgQWsfli9f7lEbRr6P+fPnw2KxoF27drracebs2bMoWbIkSpYsiRMnTnjlGkSOMPiigHDlyhWcOnUK27dvx4wZM/Dyyy+jVq1auOeee3D8+HGf9sWIDygSo0aNgsViQc2aNc3uCvmZ7OxsDBs2DID8nHhDuXLl8OyzzyI1NRVJSUleuQaRIwy+yG8tWrQIKSkpSElJwYULF3Dw4EGsWrUKb775JurXr4+cnBzMnTsXTZo0wdq1a83uLhEZ6IsvvsDOnTvRtm1b3HzzzV67zpAhQ1CiRAl89dVX2L59u9euQ2SPwRf5rejoaOuUQHx8PGrUqIGOHTti6NCh2LFjB9577z2EhoYiOTkZd955J44ePVqgjS5dukApBaUUR1eIAkRubi7Gjx8PABg0aJBXr5WQkID7778fOTk5GDdunFevRaRh8EUByWKx4KWXXsKbb74JADh9+rTXpiaIyLd+/vlnHD58GCVLlkSvXr28fr0HH3wQADBnzhwkJyd7/XpEDL4ooL388stITEwEAEyfPh1nzpzJ83xRi3+zsrLw6aefomvXrkhISEB4eDjKlCmD+vXro2fPnpg8eTLOnj1rPb9mzZqoVauW9d9du3YtsDnA/jpXrlzBzz//jIEDB6Jp06aIi4tDeHg4ypcvjxtuuAH/+9//kJmZ6fT9Pfroo7BYLOjSpQsAYPPmzXjooYdQtWpVREZGomrVqnj00Uexb9++Ir9WFy9exFtvvYVOnTqhfPny1td37NgR48aNw/79+52+dsOGDXj88cdRt25dlChRArGxsWjatClGjBiR5+vjKu37Mnr0aADAoUOHCnwdtffsSGpqKt544w00btwYJUqUQKlSpdClSxfMnj3b6Wvyry/bunUrHn/8cdSqVQuRkZEoVaqUYe/b3Z8ro96jJiUlBePHj8d1112H0qVLIyoqCtWrV8eDDz6INWvWFPn6oixYsAA33ngjypQpgxIlSqBx48Z44403kJaWprttAPj8888BAHfffTeio6MLPff48eN44YUXUKdOHev7vPPOO7Fu3TrrOWPHjoXFYsEjjzzisI3OnTujatWquHLlCr7++mtD3gNRoRSRH5kyZYoCoACo33//3aXXvPnmm9bXzJ49O89zv//+u/W5AwcO5HkuJSVFtW3b1vq8s9usWbOsr6lRo0aR59tf58UXXyzy/LZt26rz5887fG/9+vVTAFTnzp3VjBkzVGRkpMM2SpcurTZv3uz0a7R06VJVrly5QvvRq1evAq/LyclRgwcPVhaLxenrypUrp/78888iv0/27L8vzm6dO3d2eP7atWvVNddc4/R1Y8aMcXjNpKQkBUDVqFFDzZs3T0VFReV5XXx8vCHv25OfK6Peo1JKbd68WVWpUqXQaw8ZMkTl5uYWeO2BAweK/P83ePBgp+02adJEzZ071+n/OVdcvnxZhYeHKwDq+++/L/Tc3377TcXFxTnsS0hIiFq0aJFSSqmmTZsqAGru3LlO2+rfv78CoNq3b+92n4ncxeCL/IonwdfKlSutr3n55ZfzPFdY8PXqq68qACo0NFSNHDlSbdy4UZ06dUodOXJErV27Vn366aeqU6dOas6cOdbXpKamqm3btlnbXLRokUpJSclzs/9Qe/XVV1WfPn3UN998o9avX6+OHDmiTp06pf755x81atQoVaZMGQVA3X///Q7fmxZ8ValSRUVGRqpOnTqpX3/9VZ0+fVodOXJETZw4UUVERCgAql27dg7b+Ouvv6znlC5dWo0fP15t3bpVJScnq8OHD6sFCxaoJ554QvXt27fAa4cMGWJ9r4888ohauXKlOnPmjDp58qSaN2+eaty4sQKgEhIS1PHjx136fimlVHZ2tkpJSVHDhw9XAFT16tULfB3T0tIcfh9r166typUrpz788EO1f/9+dfbsWbVkyRJrX0JDQ9X27dsLXFMLvuLi4lRsbKxq1KiRmjNnjjpx4oQ6evSo+vHHHw153578XBn1Hs+ePasqVaqkAKjo6Gg1fvx4tXfvXnXmzBm1ZMkS1aFDB+s1JkyYUOD1RQVf//vf/6zPt27dWv3222/qzJkzas+ePWrUqFEqPDxc1apVS1fw9dtvv1lfv3fvXqfn7dy5U8XExCgAKjExUf3222/q9OnTavXq1apFixbW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jh07Zu2/t9xwww3WxdevvPJKoWWI8q85e+mllwDIVMzjjz/udKOAxpMRPG2U6cyZM4WuefMlve/b058rvUJDQ61pR+bOnWvdtWkvKysLL7zwAgAgKioKDz/8sMvt33777dbv17Bhwxxuavj66691b1a5/vrrAcjXf8+ePQ7PiYmJQY8ePQAAX375JQ4fPpzn+Z9++glbt261/tuVXZ1//fUXABmtvO666zzqO5GrGHyR30pPT7du67948SIOHz6MNWvW4O2330ajRo0waNAgZGdno2zZsvjxxx/dzqvUsGFD3HDDDfj444/x999/4/Tp0zh79iz++ecfDBkyBB9//DEAoFevXnnWmMTExFh3dH3wwQfYtGkT0tLSkJ2dnSeAuOeeexAaGors7Gz06NED8+bNw4kTJ3D06FFMnToVbdu2xfnz51GjRg0DvlqOWSwWTJkyBREREThz5gzatGmDCRMmYMeOHbhw4QKOHj1qrT3Zr1+/PK9t06aNNTicPXs2rrvuOnz77bc4cOAALly4gGPHjmH58uUYM2YMGjVqhMGDB7vdv5YtWwKQkbnXX38dx48fR1ZWFrKzs00bDdP7vj39uTLCa6+9hkqVKkEphTvvvBMTJkzA/v37ce7cOSxbtgzdu3fHypUrAQCjR492a3otKirKWsh+06ZN6Nq1K5YsWYJz585h3759eOONN/DEE0/kqX3qiXbt2llHtezrM+b36quvIiQkBCkpKejWrRt+/vlnnD17FgsWLMDjjz8OALjlllsQEhKC9evX4+2330ZKSorTnyvtWq1atSqyniSRbt7O4krkDvsM90XdQkND1T333KOOHTvmtL3CMty7co3mzZurkydPFmj3s88+c/oa++tMmDDB6XlRUVFq9uzZqnPnzgqA6tevX4Hr2Nd2LIxWczIpKcnh87/99psqXbp0oe/VUW3H3NxclZSUpEJDQ4v8Wt19992F9tGZ9u3bO2zPWW3HwrKmF5Yx39Vs+krpe9+e/lwZ8R6Vcq2248svv+yV2o6NGzfWXdtRKaXuuOMOBUA99NBDhZ734YcfOq2/WblyZXXs2DF133335Xl83rx5BdrJzc1V1apVUwDUe++951GfidzBkS8KCJGRkUhISECDBg1w33334d1338WBAwcwe/bsInNgOfP3339jwoQJuPXWW5GYmIi4uDiEh4ejQoUKuOmmm/C///0P69atQ4UKFQq8dsCAAfjuu+/QpUsXlClTBiEhjv8r/ec//8HChQvRvXt3xMXFITIyEjVr1kT//v2xfv163HPPPR713V033HCDdXSiTZs2KF26NCIiIlCtWjV07NgR48ePd7j7zmKxYNSoUdixYwdeeuklNGvWDPHx8dYps2bNmuGZZ57BkiVLMHPmTI/6tmjRIvznP/9Bw4YNDR8J8pSe963n58oITZo0wY4dOzB27Fi0bt0a8fHx1u/1Aw88gNWrV+Odd95xKyGxvXfffRc//fQTunfvjlKlSllHgl977TX8+eefbuXac+app54CAMybNy9PWon8nn32Wfzxxx+49957UblyZYSFhSEmJgYdOnTAsmXLULlyZXz11VcYOHBgoTs7V6xYgSNHjiAyMrLACDCRN1iU8kE9ECIiIhfl5uaidu3aOHToEL799lv07dvXq9d78skn8cUXX6Bv3766cvsRuYojX0RE5FdCQkKsdTU/+OADr17r7NmzmDFjBkJDQ631Oom8jcEXERH5nSeeeAINGjTAn3/+icWLF3vtOm+//TYuX76Mxx57DI0bN/badYjscdqRiIj80k8//YRevXqhbdu2WLt2reHtnz17FrVq1YJSCnv27HGrPBmRHgy+iIiIiHyI045EREREPsTgi4iIiMiHGHwRERER+RCDLyIiIiIfYvBFRERE5EMMvoiIiIh8iMEXERERkQ8x+CIiIiLyIQZfRERERD7E4IuIiIjIhxh8EREREfnQ/wMjyk65BSPPqgAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Selected alpha: 0.3\n",
      "Round 1.5\n",
      "\n",
      "Adjacency Matrix\n",
      "[[0.0000 0.0431 1.0000 ... 0.0133 1.0000 1.0000]\n",
      " [0.0431 0.0000 1.0000 ... 0.0348 1.0000 1.0000]\n",
      " [1.0000 1.0000 0.0000 ... 1.0000 1.0000 1.0000]\n",
      " ...\n",
      " [0.0133 0.0348 1.0000 ... 0.0000 1.0000 1.0000]\n",
      " [1.0000 1.0000 1.0000 ... 1.0000 0.0000 0.0082]\n",
      " [1.0000 1.0000 1.0000 ... 1.0000 0.0082 0.0000]]\n",
      "\n",
      "Clusters: \n",
      "[[0, 4, 43, 73, 87, 63, 22, 53, 83, 69, 97, 29, 41, 17, 86, 59, 9, 37, 65, 19, 1, 7], [2, 24, 56, 33, 91, 23, 92, 28, 48, 81, 93, 13, 18, 44, 16, 6, 25, 76, 32, 50, 64, 38, 10, 88], [3, 5, 46, 85, 79, 84, 94, 99, 95, 72, 98, 12, 11, 62, 45, 27, 36, 80, 68], [8, 40, 55, 60, 52, 51, 54, 70, 20, 82, 49, 66, 30, 71, 31], [14, 47, 61, 90, 75, 89, 26, 42, 67, 15, 39, 21, 58, 57, 34, 77, 78, 96, 74, 35]]\n",
      "\n",
      "Number of Clusters 5\n",
      "\n",
      "Cluster 0: 22 \n",
      "Cluster 1: 24 \n",
      "Cluster 2: 19 \n",
      "Cluster 3: 15 \n",
      "Cluster 4: 20 \n",
      "Clients: Cluster_ID \n",
      "{0: 0, 1: 0, 2: 1, 3: 2, 4: 0, 5: 2, 6: 1, 7: 0, 8: 3, 9: 0, 10: 1, 11: 2, 12: 2, 13: 1, 14: 4, 15: 4, 16: 1, 17: 0, 18: 1, 19: 0, 20: 3, 21: 4, 22: 0, 23: 1, 24: 1, 25: 1, 26: 4, 27: 2, 28: 1, 29: 0, 30: 3, 31: 3, 32: 1, 33: 1, 34: 4, 35: 4, 36: 2, 37: 0, 38: 1, 39: 4, 40: 3, 41: 0, 42: 4, 43: 0, 44: 1, 45: 2, 46: 2, 47: 4, 48: 1, 49: 3, 50: 1, 51: 3, 52: 3, 53: 0, 54: 3, 55: 3, 56: 1, 57: 4, 58: 4, 59: 0, 60: 3, 61: 4, 62: 2, 63: 0, 64: 1, 65: 0, 66: 3, 67: 4, 68: 2, 69: 0, 70: 3, 71: 3, 72: 2, 73: 0, 74: 4, 75: 4, 76: 1, 77: 4, 78: 4, 79: 2, 80: 2, 81: 1, 82: 3, 83: 0, 84: 2, 85: 2, 86: 0, 87: 0, 88: 1, 89: 4, 90: 4, 91: 1, 92: 1, 93: 1, 94: 2, 95: 2, 96: 4, 97: 0, 98: 2, 99: 2}\n",
      "Cluster 0, First Client 0: {4: 2, 8: 115, 9: 119}\n",
      "Cluster 1, First Client 2: {0: 79, 1: 126, 3: 89}\n",
      "Cluster 2, First Client 3: {2: 1, 5: 475, 7: 397}\n",
      "Cluster 3, First Client 8: {1: 350, 4: 88, 7: 62}\n",
      "Cluster 4, First Client 14: {0: 54, 4: 149, 6: 188}\n"
     ]
    }
   ],
   "source": [
    "#HC 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",
    "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",
    "adj_mat=A.detach().cpu().numpy()\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 # 0.01*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+\"_c_lda.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()\n",
    "# Find alpha with lowest loss and fewer than 8 clusters\n",
    "valid_alphas = [\n",
    "    (a, loss) for a, loss, count in zip(alpha_grid, cluster_losses, cluster_counts) if count < 8\n",
    "]\n",
    "\n",
    "if valid_alphas:\n",
    "    best_alpha = min(valid_alphas, key=lambda x: x[1])[0]\n",
    "    args.cluster_alpha = best_alpha\n",
    "    print(f\"Selected alpha: {args.cluster_alpha}\")\n",
    "else:\n",
    "    print(\"No alpha found with cluster count < 8\")\n",
    "\n",
    "\n",
    "args.cluster_alpha=.1\n",
    "\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",
    "\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='minimum')\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",
    "    \n",
    "similarity_graph = build_similarity_graph_rank_supply_nonzero(\n",
    "    traindata_cls_counts,\n",
    "    clusters,\n",
    "    num_classes=10,\n",
    "    top_k=1\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": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      ">>> All clusters switched to combined \n",
      "\n",
      "##### ROUND 1 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.4425\n",
      "avg best  : 15.79%\n",
      "\n",
      "Client   0 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   1 | best_pre=  0.07 | best_post= 79.57\n",
      "Client   2 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   3 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   4 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   5 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   6 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   7 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   8 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   9 | best_pre=  0.07 | best_post= 96.57\n",
      "Client  10 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  11 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  12 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  13 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  14 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  15 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  16 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  17 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  18 | best_pre= 30.97 | best_post= 70.23\n",
      "Client  19 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  20 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  21 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  22 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  23 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  24 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  25 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  26 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  27 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  28 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  29 | best_pre=  0.07 | best_post= 95.90\n",
      "Client  30 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  31 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  32 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  33 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  34 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  35 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  36 | best_pre=  0.00 | best_post= 91.10\n",
      "Client  37 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  38 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  39 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  40 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  41 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  42 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  43 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  44 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  45 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  46 | best_pre=  0.00 | best_post= 97.27\n",
      "Client  47 | best_pre=  0.17 | best_post= 51.87\n",
      "Client  48 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  49 | best_pre= 32.03 | best_post= 96.03\n",
      "Client  50 | best_pre= 30.97 | best_post= 88.53\n",
      "Client  51 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  52 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  53 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  54 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  55 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  56 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  57 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  58 | best_pre=  0.17 | best_post= 40.77\n",
      "Client  59 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  60 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  61 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  62 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  63 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  64 | best_pre= 30.97 | best_post= 58.70\n",
      "Client  65 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  66 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  67 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  68 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  69 | best_pre=  0.07 | best_post= 94.73\n",
      "Client  70 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  71 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  72 | best_pre=  0.00 | best_post= 88.33\n",
      "Client  73 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  74 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  76 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  77 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  78 | best_pre=  0.17 | best_post= 33.33\n",
      "Client  79 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  80 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  81 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  82 | best_pre= 32.03 | best_post= 95.93\n",
      "Client  83 | best_pre=  0.07 | best_post= 92.83\n",
      "Client  84 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  85 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  86 | best_pre=  0.07 | best_post= 66.47\n",
      "Client  87 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  88 | best_pre= 30.97 | best_post= 56.07\n",
      "Client  89 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  90 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  91 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  92 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  93 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  94 | best_pre=  0.00 | best_post= 90.73\n",
      "Client  95 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  96 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  97 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  98 | best_pre=  0.00 | best_post= 94.10\n",
      "Client  99 | best_pre=  0.00 | best_post=  0.00\n",
      "\n",
      "##### ROUND 2 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.1668\n",
      "avg best  : 28.46%\n",
      "\n",
      "Client   0 | best_pre= 84.20 | best_post= 94.57\n",
      "Client   1 | best_pre=  0.07 | best_post= 79.57\n",
      "Client   2 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   3 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   4 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   5 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   6 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   7 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   8 | best_pre= 96.30 | best_post= 98.87\n",
      "Client   9 | best_pre=  0.07 | best_post= 96.57\n",
      "Client  10 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  11 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  12 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  13 | best_pre= 59.03 | best_post= 89.57\n",
      "Client  14 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  15 | best_pre= 33.33 | best_post= 56.57\n",
      "Client  16 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  17 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  18 | best_pre= 30.97 | best_post= 70.23\n",
      "Client  19 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  20 | best_pre= 96.30 | best_post= 96.07\n",
      "Client  21 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  22 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  23 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  24 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  25 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  26 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  27 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  28 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  29 | best_pre=  0.07 | best_post= 95.90\n",
      "Client  30 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  31 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  32 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  33 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  34 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  35 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  36 | best_pre=  0.00 | best_post= 91.10\n",
      "Client  37 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  38 | best_pre= 59.03 | best_post= 84.50\n",
      "Client  39 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  40 | best_pre= 96.30 | best_post= 99.10\n",
      "Client  41 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  42 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  43 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  44 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  45 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  46 | best_pre=  0.00 | best_post= 97.27\n",
      "Client  47 | best_pre= 33.33 | best_post= 72.47\n",
      "Client  48 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  49 | best_pre= 32.03 | best_post= 96.03\n",
      "Client  50 | best_pre= 59.03 | best_post= 88.53\n",
      "Client  51 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  52 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  53 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  54 | best_pre= 96.30 | best_post= 97.13\n",
      "Client  55 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  56 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  57 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  58 | best_pre= 33.33 | best_post= 68.10\n",
      "Client  59 | best_pre= 84.20 | best_post= 96.97\n",
      "Client  60 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  61 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  62 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  63 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  64 | best_pre= 59.03 | best_post= 76.60\n",
      "Client  65 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  66 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  67 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  68 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  69 | best_pre= 84.20 | best_post= 97.47\n",
      "Client  70 | best_pre= 96.30 | best_post= 96.87\n",
      "Client  71 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  72 | best_pre=  0.00 | best_post= 88.33\n",
      "Client  73 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  74 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  76 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  77 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  78 | best_pre= 33.33 | best_post= 33.33\n",
      "Client  79 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  80 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  81 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  82 | best_pre= 32.03 | best_post= 95.93\n",
      "Client  83 | best_pre=  0.07 | best_post= 92.83\n",
      "Client  84 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  85 | best_pre= 95.93 | best_post= 97.77\n",
      "Client  86 | best_pre= 84.20 | best_post= 93.67\n",
      "Client  87 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  88 | best_pre= 30.97 | best_post= 56.07\n",
      "Client  89 | best_pre= 33.33 | best_post= 65.43\n",
      "Client  90 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  91 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  92 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  93 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  94 | best_pre=  0.00 | best_post= 90.73\n",
      "Client  95 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  96 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  97 | best_pre= 84.20 | best_post= 97.07\n",
      "Client  98 | best_pre=  0.00 | best_post= 94.10\n",
      "Client  99 | best_pre=  0.00 | best_post=  0.00\n",
      "\n",
      "##### ROUND 3 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.1197\n",
      "avg best  : 43.12%\n",
      "\n",
      "Client   0 | best_pre= 97.63 | best_post= 94.57\n",
      "Client   1 | best_pre= 97.63 | best_post= 79.57\n",
      "Client   2 | best_pre= 86.70 | best_post= 90.90\n",
      "Client   3 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   4 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   5 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   6 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   7 | best_pre= 97.63 | best_post= 97.60\n",
      "Client   8 | best_pre= 96.30 | best_post= 98.87\n",
      "Client   9 | best_pre=  0.07 | best_post= 96.57\n",
      "Client  10 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  11 | best_pre= 97.63 | best_post= 97.50\n",
      "Client  12 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  13 | best_pre= 86.70 | best_post= 89.63\n",
      "Client  14 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  15 | best_pre= 33.33 | best_post= 56.57\n",
      "Client  16 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  17 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  18 | best_pre= 30.97 | best_post= 70.23\n",
      "Client  19 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  20 | best_pre= 96.30 | best_post= 96.07\n",
      "Client  21 | best_pre= 71.63 | best_post= 51.47\n",
      "Client  22 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  23 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  24 | best_pre= 86.70 | best_post= 93.63\n",
      "Client  25 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  26 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  27 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  28 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  29 | best_pre=  0.07 | best_post= 95.90\n",
      "Client  30 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  31 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  32 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  33 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  34 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  35 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  36 | best_pre=  0.00 | best_post= 91.10\n",
      "Client  37 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  38 | best_pre= 59.03 | best_post= 84.50\n",
      "Client  39 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  40 | best_pre= 96.30 | best_post= 99.10\n",
      "Client  41 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  42 | best_pre= 71.63 | best_post= 65.43\n",
      "Client  43 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  44 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  45 | best_pre= 97.63 | best_post= 95.83\n",
      "Client  46 | best_pre=  0.00 | best_post= 97.27\n",
      "Client  47 | best_pre= 33.33 | best_post= 72.47\n",
      "Client  48 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  49 | best_pre= 32.03 | best_post= 96.03\n",
      "Client  50 | best_pre= 59.03 | best_post= 88.53\n",
      "Client  51 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  52 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  53 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  54 | best_pre= 96.30 | best_post= 97.13\n",
      "Client  55 | best_pre= 98.57 | best_post= 97.77\n",
      "Client  56 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  57 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  58 | best_pre= 33.33 | best_post= 68.10\n",
      "Client  59 | best_pre= 84.20 | best_post= 96.97\n",
      "Client  60 | best_pre= 98.57 | best_post= 98.60\n",
      "Client  61 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  62 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  63 | best_pre= 97.63 | best_post= 97.43\n",
      "Client  64 | best_pre= 59.03 | best_post= 76.60\n",
      "Client  65 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  66 | best_pre= 98.57 | best_post= 99.00\n",
      "Client  67 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  68 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  69 | best_pre= 84.20 | best_post= 97.47\n",
      "Client  70 | best_pre= 96.30 | best_post= 96.87\n",
      "Client  71 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  72 | best_pre= 97.63 | best_post= 96.87\n",
      "Client  73 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  74 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  76 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  77 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  78 | best_pre= 33.33 | best_post= 33.33\n",
      "Client  79 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  80 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  81 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  82 | best_pre= 32.03 | best_post= 95.93\n",
      "Client  83 | best_pre=  0.07 | best_post= 92.83\n",
      "Client  84 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  85 | best_pre= 95.93 | best_post= 97.77\n",
      "Client  86 | best_pre= 84.20 | best_post= 93.67\n",
      "Client  87 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  88 | best_pre= 30.97 | best_post= 56.07\n",
      "Client  89 | best_pre= 33.33 | best_post= 65.43\n",
      "Client  90 | best_pre= 71.63 | best_post= 70.60\n",
      "Client  91 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  92 | best_pre= 86.70 | best_post= 90.37\n",
      "Client  93 | best_pre= 86.70 | best_post= 90.10\n",
      "Client  94 | best_pre=  0.00 | best_post= 90.73\n",
      "Client  95 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  96 | best_pre= 71.63 | best_post= 68.57\n",
      "Client  97 | best_pre= 84.20 | best_post= 97.07\n",
      "Client  98 | best_pre=  0.00 | best_post= 94.10\n",
      "Client  99 | best_pre= 97.63 | best_post= 97.73\n",
      "\n",
      "##### ROUND 4 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.0654\n",
      "avg best  : 51.52%\n",
      "\n",
      "Client   0 | best_pre= 97.63 | best_post= 94.57\n",
      "Client   1 | best_pre= 97.63 | best_post= 79.57\n",
      "Client   2 | best_pre= 86.70 | best_post= 90.90\n",
      "Client   3 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   4 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   5 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   6 | best_pre=  0.00 | best_post=  0.00\n",
      "Client   7 | best_pre= 97.63 | best_post= 97.60\n",
      "Client   8 | best_pre= 99.17 | best_post= 99.17\n",
      "Client   9 | best_pre=  0.07 | best_post= 96.57\n",
      "Client  10 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  11 | best_pre= 97.63 | best_post= 97.50\n",
      "Client  12 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  13 | best_pre= 86.70 | best_post= 89.63\n",
      "Client  14 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  15 | best_pre= 33.33 | best_post= 56.57\n",
      "Client  16 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  17 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  18 | best_pre= 30.97 | best_post= 70.23\n",
      "Client  19 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  20 | best_pre= 96.30 | best_post= 96.07\n",
      "Client  21 | best_pre= 71.63 | best_post= 51.47\n",
      "Client  22 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  23 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  24 | best_pre= 86.70 | best_post= 93.63\n",
      "Client  25 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  26 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  27 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  28 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  29 | best_pre=  0.07 | best_post= 95.90\n",
      "Client  30 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  31 | best_pre= 99.17 | best_post= 97.50\n",
      "Client  32 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  33 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  34 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  35 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  36 | best_pre= 98.03 | best_post= 97.37\n",
      "Client  37 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  38 | best_pre= 59.03 | best_post= 84.50\n",
      "Client  39 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  40 | best_pre= 96.30 | best_post= 99.10\n",
      "Client  41 | best_pre= 97.60 | best_post= 97.20\n",
      "Client  42 | best_pre= 71.63 | best_post= 65.43\n",
      "Client  43 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  44 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  45 | best_pre= 98.03 | best_post= 95.83\n",
      "Client  46 | best_pre=  0.00 | best_post= 97.27\n",
      "Client  47 | best_pre= 33.33 | best_post= 72.47\n",
      "Client  48 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  49 | best_pre= 32.03 | best_post= 96.03\n",
      "Client  50 | best_pre= 59.03 | best_post= 88.53\n",
      "Client  51 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  52 | best_pre= 99.17 | best_post= 98.90\n",
      "Client  53 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  54 | best_pre= 96.30 | best_post= 97.13\n",
      "Client  55 | best_pre= 98.57 | best_post= 97.77\n",
      "Client  56 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  57 | best_pre= 72.73 | best_post= 66.37\n",
      "Client  58 | best_pre= 72.73 | best_post= 68.10\n",
      "Client  59 | best_pre= 84.20 | best_post= 96.97\n",
      "Client  60 | best_pre= 98.57 | best_post= 98.60\n",
      "Client  61 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  62 | best_pre= 98.03 | best_post= 96.77\n",
      "Client  63 | best_pre= 97.63 | best_post= 97.43\n",
      "Client  64 | best_pre= 92.93 | best_post= 85.70\n",
      "Client  65 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  66 | best_pre= 98.57 | best_post= 99.00\n",
      "Client  67 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  68 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  69 | best_pre= 97.60 | best_post= 98.60\n",
      "Client  70 | best_pre= 99.17 | best_post= 99.07\n",
      "Client  71 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  72 | best_pre= 97.63 | best_post= 96.87\n",
      "Client  73 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  74 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  76 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  77 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  78 | best_pre= 72.73 | best_post= 33.33\n",
      "Client  79 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  80 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  81 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  82 | best_pre= 32.03 | best_post= 95.93\n",
      "Client  83 | best_pre= 97.60 | best_post= 97.70\n",
      "Client  84 | best_pre= 98.03 | best_post= 97.83\n",
      "Client  85 | best_pre= 98.03 | best_post= 97.97\n",
      "Client  86 | best_pre= 97.60 | best_post= 93.67\n",
      "Client  87 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  88 | best_pre= 30.97 | best_post= 56.07\n",
      "Client  89 | best_pre= 33.33 | best_post= 65.43\n",
      "Client  90 | best_pre= 71.63 | best_post= 70.60\n",
      "Client  91 | best_pre= 92.93 | best_post= 94.17\n",
      "Client  92 | best_pre= 86.70 | best_post= 90.37\n",
      "Client  93 | best_pre= 92.93 | best_post= 90.73\n",
      "Client  94 | best_pre=  0.00 | best_post= 90.73\n",
      "Client  95 | best_pre= 98.03 | best_post= 97.67\n",
      "Client  96 | best_pre= 71.63 | best_post= 68.57\n",
      "Client  97 | best_pre= 84.20 | best_post= 97.07\n",
      "Client  98 | best_pre=  0.00 | best_post= 94.10\n",
      "Client  99 | best_pre= 97.63 | best_post= 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.0511\n",
      "avg best  : 85.19%\n",
      "\n",
      "Client   0 | best_pre= 97.63 | best_post= 95.53\n",
      "Client   1 | best_pre= 97.63 | best_post= 79.57\n",
      "Client   2 | best_pre= 94.90 | best_post= 93.30\n",
      "Client   3 | best_pre= 98.60 | best_post= 97.93\n",
      "Client   4 | best_pre= 98.53 | best_post= 98.30\n",
      "Client   5 | best_pre= 98.60 | best_post= 98.50\n",
      "Client   6 | best_pre= 94.40 | best_post= 94.97\n",
      "Client   7 | best_pre= 97.63 | best_post= 97.60\n",
      "Client   8 | best_pre= 99.17 | best_post= 99.17\n",
      "Client   9 | best_pre= 98.57 | best_post= 98.30\n",
      "Client  10 | best_pre= 95.87 | best_post= 95.77\n",
      "Client  11 | best_pre= 98.27 | best_post= 98.20\n",
      "Client  12 | best_pre= 98.27 | best_post= 95.73\n",
      "Client  13 | best_pre= 92.87 | best_post= 93.17\n",
      "Client  14 | best_pre= 49.43 | best_post= 71.40\n",
      "Client  15 | best_pre= 76.57 | best_post= 78.50\n",
      "Client  16 | best_pre= 95.87 | best_post= 93.27\n",
      "Client  17 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  18 | best_pre= 95.93 | best_post= 95.17\n",
      "Client  19 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  20 | best_pre= 99.13 | best_post= 98.70\n",
      "Client  21 | best_pre= 80.07 | best_post= 61.23\n",
      "Client  22 | best_pre= 98.57 | best_post= 98.37\n",
      "Client  23 | best_pre= 94.90 | best_post= 89.23\n",
      "Client  24 | best_pre= 94.40 | best_post= 93.83\n",
      "Client  25 | best_pre= 95.93 | best_post= 95.13\n",
      "Client  26 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  27 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  28 | best_pre= 94.90 | best_post= 91.07\n",
      "Client  29 | best_pre=  0.07 | best_post= 95.90\n",
      "Client  30 | best_pre= 99.27 | best_post= 99.13\n",
      "Client  31 | best_pre= 99.40 | best_post= 98.13\n",
      "Client  32 | best_pre= 92.87 | best_post= 85.83\n",
      "Client  33 | best_pre= 94.90 | best_post= 93.93\n",
      "Client  34 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  35 | best_pre= 53.40 | best_post= 65.37\n",
      "Client  36 | best_pre= 98.73 | best_post= 98.03\n",
      "Client  37 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  38 | best_pre= 92.87 | best_post= 93.10\n",
      "Client  39 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  40 | best_pre= 99.40 | best_post= 99.27\n",
      "Client  41 | best_pre= 98.57 | best_post= 98.13\n",
      "Client  42 | best_pre= 71.63 | best_post= 65.43\n",
      "Client  43 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  44 | best_pre= 95.93 | best_post= 94.40\n",
      "Client  45 | best_pre= 98.03 | best_post= 95.83\n",
      "Client  46 | best_pre= 98.63 | best_post= 98.37\n",
      "Client  47 | best_pre= 49.43 | best_post= 78.97\n",
      "Client  48 | best_pre= 95.93 | best_post= 94.50\n",
      "Client  49 | best_pre= 99.33 | best_post= 99.10\n",
      "Client  50 | best_pre= 95.87 | best_post= 92.07\n",
      "Client  51 | best_pre= 99.30 | best_post= 98.83\n",
      "Client  52 | best_pre= 99.40 | best_post= 99.03\n",
      "Client  53 | best_pre= 98.57 | best_post= 98.30\n",
      "Client  54 | best_pre= 99.30 | best_post= 99.33\n",
      "Client  55 | best_pre= 99.27 | best_post= 99.37\n",
      "Client  56 | best_pre= 94.90 | best_post= 93.30\n",
      "Client  57 | best_pre= 72.73 | best_post= 66.37\n",
      "Client  58 | best_pre= 80.07 | best_post= 72.00\n",
      "Client  59 | best_pre= 98.57 | best_post= 98.27\n",
      "Client  60 | best_pre= 99.30 | best_post= 99.43\n",
      "Client  61 | best_pre= 53.40 | best_post= 73.83\n",
      "Client  62 | best_pre= 98.07 | best_post= 97.67\n",
      "Client  63 | best_pre= 97.83 | best_post= 98.73\n",
      "Client  64 | best_pre= 94.93 | best_post= 93.87\n",
      "Client  65 | best_pre= 97.83 | best_post= 98.50\n",
      "Client  66 | best_pre= 99.27 | best_post= 99.17\n",
      "Client  67 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  68 | best_pre= 98.73 | best_post= 96.13\n",
      "Client  69 | best_pre= 98.47 | best_post= 98.60\n",
      "Client  70 | best_pre= 99.40 | best_post= 99.37\n",
      "Client  71 | best_pre= 99.33 | best_post= 99.27\n",
      "Client  72 | best_pre= 98.60 | best_post= 98.00\n",
      "Client  73 | best_pre= 98.87 | best_post= 98.57\n",
      "Client  74 | best_pre= 74.67 | best_post= 34.23\n",
      "Client  75 | best_pre= 74.67 | best_post= 75.67\n",
      "Client  76 | best_pre= 95.93 | best_post= 94.63\n",
      "Client  77 | best_pre= 72.70 | best_post= 54.07\n",
      "Client  78 | best_pre= 72.73 | best_post= 69.03\n",
      "Client  79 | best_pre= 98.23 | best_post= 98.33\n",
      "Client  80 | best_pre= 98.60 | best_post= 98.53\n",
      "Client  81 | best_pre= 95.93 | best_post= 94.97\n",
      "Client  82 | best_pre= 99.40 | best_post= 99.10\n",
      "Client  83 | best_pre= 98.87 | best_post= 97.73\n",
      "Client  84 | best_pre= 98.03 | best_post= 97.83\n",
      "Client  85 | best_pre= 98.60 | best_post= 98.63\n",
      "Client  86 | best_pre= 98.87 | best_post= 96.27\n",
      "Client  87 | best_pre= 97.83 | best_post= 98.63\n",
      "Client  88 | best_pre= 95.80 | best_post= 92.40\n",
      "Client  89 | best_pre= 64.67 | best_post= 76.40\n",
      "Client  90 | best_pre= 74.67 | best_post= 78.40\n",
      "Client  91 | best_pre= 95.87 | best_post= 95.07\n",
      "Client  92 | best_pre= 95.87 | best_post= 94.27\n",
      "Client  93 | best_pre= 95.93 | best_post= 93.77\n",
      "Client  94 | best_pre= 98.07 | best_post= 97.73\n",
      "Client  95 | best_pre= 98.03 | best_post= 97.67\n",
      "Client  96 | best_pre= 80.07 | best_post= 77.40\n",
      "Client  97 | best_pre= 98.87 | best_post= 98.10\n",
      "Client  98 | best_pre= 98.63 | best_post= 98.03\n",
      "Client  99 | best_pre= 98.73 | best_post= 98.47\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.0234\n",
      "avg best  : 91.58%\n",
      "\n",
      "Client   0 | best_pre= 99.03 | best_post= 98.23\n",
      "Client   1 | best_pre= 98.60 | best_post= 93.37\n",
      "Client   2 | best_pre= 96.10 | best_post= 96.00\n",
      "Client   3 | best_pre= 98.73 | best_post= 98.57\n",
      "Client   4 | best_pre= 98.97 | best_post= 98.87\n",
      "Client   5 | best_pre= 98.70 | best_post= 98.50\n",
      "Client   6 | best_pre= 95.63 | best_post= 96.10\n",
      "Client   7 | best_pre= 99.10 | best_post= 98.53\n",
      "Client   8 | best_pre= 99.43 | best_post= 99.53\n",
      "Client   9 | best_pre= 99.10 | best_post= 98.83\n",
      "Client  10 | best_pre= 95.87 | best_post= 95.77\n",
      "Client  11 | best_pre= 98.73 | best_post= 98.53\n",
      "Client  12 | best_pre= 98.73 | best_post= 96.73\n",
      "Client  13 | best_pre= 95.63 | best_post= 94.40\n",
      "Client  14 | best_pre= 67.23 | best_post= 77.37\n",
      "Client  15 | best_pre= 76.57 | best_post= 79.70\n",
      "Client  16 | best_pre= 95.87 | best_post= 93.27\n",
      "Client  17 | best_pre= 98.73 | best_post= 97.83\n",
      "Client  18 | best_pre= 95.93 | best_post= 95.17\n",
      "Client  19 | best_pre= 99.03 | best_post= 99.03\n",
      "Client  20 | best_pre= 99.47 | best_post= 98.90\n",
      "Client  21 | best_pre= 80.07 | best_post= 69.50\n",
      "Client  22 | best_pre= 98.57 | best_post= 98.37\n",
      "Client  23 | best_pre= 94.90 | best_post= 93.70\n",
      "Client  24 | best_pre= 95.63 | best_post= 95.07\n",
      "Client  25 | best_pre= 95.97 | best_post= 95.90\n",
      "Client  26 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  27 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  28 | best_pre= 96.10 | best_post= 91.97\n",
      "Client  29 | best_pre= 99.10 | best_post= 98.97\n",
      "Client  30 | best_pre= 99.47 | best_post= 99.47\n",
      "Client  31 | best_pre= 99.53 | best_post= 99.47\n",
      "Client  32 | best_pre= 95.83 | best_post= 89.70\n",
      "Client  33 | best_pre= 95.83 | best_post= 95.07\n",
      "Client  34 | best_pre= 81.30 | best_post= 74.37\n",
      "Client  35 | best_pre= 81.70 | best_post= 70.67\n",
      "Client  36 | best_pre= 98.73 | best_post= 98.40\n",
      "Client  37 | best_pre= 99.10 | best_post= 98.83\n",
      "Client  38 | best_pre= 95.97 | best_post= 94.27\n",
      "Client  39 | best_pre=  0.00 | best_post=  0.00\n",
      "Client  40 | best_pre= 99.40 | best_post= 99.27\n",
      "Client  41 | best_pre= 99.10 | best_post= 98.13\n",
      "Client  42 | best_pre= 75.63 | best_post= 75.27\n",
      "Client  43 | best_pre= 99.10 | best_post= 98.27\n",
      "Client  44 | best_pre= 95.93 | best_post= 95.27\n",
      "Client  45 | best_pre= 98.03 | best_post= 95.83\n",
      "Client  46 | best_pre= 98.73 | best_post= 98.67\n",
      "Client  47 | best_pre= 74.37 | best_post= 80.63\n",
      "Client  48 | best_pre= 95.97 | best_post= 95.60\n",
      "Client  49 | best_pre= 99.47 | best_post= 99.40\n",
      "Client  50 | best_pre= 95.87 | best_post= 92.07\n",
      "Client  51 | best_pre= 99.37 | best_post= 99.50\n",
      "Client  52 | best_pre= 99.53 | best_post= 99.20\n",
      "Client  53 | best_pre= 98.97 | best_post= 98.73\n",
      "Client  54 | best_pre= 99.47 | best_post= 99.50\n",
      "Client  55 | best_pre= 99.53 | best_post= 99.47\n",
      "Client  56 | best_pre= 96.10 | best_post= 94.37\n",
      "Client  57 | best_pre= 72.73 | best_post= 66.37\n",
      "Client  58 | best_pre= 80.07 | best_post= 72.00\n",
      "Client  59 | best_pre= 99.10 | best_post= 98.80\n",
      "Client  60 | best_pre= 99.47 | best_post= 99.43\n",
      "Client  61 | best_pre= 53.40 | best_post= 73.83\n",
      "Client  62 | best_pre= 98.73 | best_post= 98.47\n",
      "Client  63 | best_pre= 98.60 | best_post= 99.13\n",
      "Client  64 | best_pre= 95.97 | best_post= 93.87\n",
      "Client  65 | best_pre= 98.73 | best_post= 98.90\n",
      "Client  66 | best_pre= 99.27 | best_post= 99.23\n",
      "Client  67 | best_pre= 68.80 | best_post= 75.50\n",
      "Client  68 | best_pre= 98.73 | best_post= 96.13\n",
      "Client  69 | best_pre= 98.73 | best_post= 98.90\n",
      "Client  70 | best_pre= 99.40 | best_post= 99.50\n",
      "Client  71 | best_pre= 99.47 | best_post= 99.40\n",
      "Client  72 | best_pre= 98.63 | best_post= 98.00\n",
      "Client  73 | best_pre= 99.03 | best_post= 99.03\n",
      "Client  74 | best_pre= 74.67 | best_post= 34.23\n",
      "Client  75 | best_pre= 78.07 | best_post= 76.17\n",
      "Client  76 | best_pre= 96.10 | best_post= 95.80\n",
      "Client  77 | best_pre= 81.70 | best_post= 59.10\n",
      "Client  78 | best_pre= 81.70 | best_post= 69.03\n",
      "Client  79 | best_pre= 98.73 | best_post= 98.57\n",
      "Client  80 | best_pre= 98.73 | best_post= 98.63\n",
      "Client  81 | best_pre= 95.93 | best_post= 94.97\n",
      "Client  82 | best_pre= 99.40 | best_post= 99.10\n",
      "Client  83 | best_pre= 99.10 | best_post= 97.80\n",
      "Client  84 | best_pre= 98.70 | best_post= 98.43\n",
      "Client  85 | best_pre= 98.60 | best_post= 98.63\n",
      "Client  86 | best_pre= 98.87 | best_post= 96.27\n",
      "Client  87 | best_pre= 98.97 | best_post= 98.90\n",
      "Client  88 | best_pre= 95.83 | best_post= 93.60\n",
      "Client  89 | best_pre= 74.37 | best_post= 76.93\n",
      "Client  90 | best_pre= 75.63 | best_post= 78.40\n",
      "Client  91 | best_pre= 95.87 | best_post= 95.50\n",
      "Client  92 | best_pre= 95.87 | best_post= 95.37\n",
      "Client  93 | best_pre= 95.93 | best_post= 95.07\n",
      "Client  94 | best_pre= 98.73 | best_post= 98.17\n",
      "Client  95 | best_pre= 98.67 | best_post= 98.33\n",
      "Client  96 | best_pre= 81.70 | best_post= 78.80\n",
      "Client  97 | best_pre= 98.97 | best_post= 98.23\n",
      "Client  98 | best_pre= 98.63 | best_post= 98.10\n",
      "Client  99 | best_pre= 98.73 | best_post= 98.63\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.0190\n",
      "avg best  : 94.96%\n",
      "\n",
      "Client   0 | best_pre= 99.20 | best_post= 98.53\n",
      "Client   1 | best_pre= 99.17 | best_post= 97.70\n",
      "Client   2 | best_pre= 96.10 | best_post= 96.00\n",
      "Client   3 | best_pre= 98.73 | best_post= 98.80\n",
      "Client   4 | best_pre= 98.97 | best_post= 99.03\n",
      "Client   5 | best_pre= 98.70 | best_post= 98.70\n",
      "Client   6 | best_pre= 96.23 | best_post= 96.17\n",
      "Client   7 | best_pre= 99.20 | best_post= 98.53\n",
      "Client   8 | best_pre= 99.53 | best_post= 99.57\n",
      "Client   9 | best_pre= 99.10 | best_post= 99.07\n",
      "Client  10 | best_pre= 96.70 | best_post= 96.77\n",
      "Client  11 | best_pre= 98.73 | best_post= 98.70\n",
      "Client  12 | best_pre= 98.73 | best_post= 96.73\n",
      "Client  13 | best_pre= 96.37 | best_post= 94.53\n",
      "Client  14 | best_pre= 82.47 | best_post= 80.40\n",
      "Client  15 | best_pre= 82.93 | best_post= 80.63\n",
      "Client  16 | best_pre= 96.17 | best_post= 94.27\n",
      "Client  17 | best_pre= 99.10 | best_post= 97.83\n",
      "Client  18 | best_pre= 96.50 | best_post= 96.17\n",
      "Client  19 | best_pre= 99.03 | best_post= 99.07\n",
      "Client  20 | best_pre= 99.60 | best_post= 99.03\n",
      "Client  21 | best_pre= 82.47 | best_post= 69.50\n",
      "Client  22 | best_pre= 99.20 | best_post= 98.97\n",
      "Client  23 | best_pre= 96.57 | best_post= 93.87\n",
      "Client  24 | best_pre= 96.57 | best_post= 95.63\n",
      "Client  25 | best_pre= 96.70 | best_post= 96.37\n",
      "Client  26 | best_pre= 82.47 | best_post= 78.50\n",
      "Client  27 | best_pre= 98.67 | best_post= 98.03\n",
      "Client  28 | best_pre= 96.10 | best_post= 91.97\n",
      "Client  29 | best_pre= 99.10 | best_post= 99.17\n",
      "Client  30 | best_pre= 99.60 | best_post= 99.57\n",
      "Client  31 | best_pre= 99.60 | best_post= 99.60\n",
      "Client  32 | best_pre= 96.43 | best_post= 89.70\n",
      "Client  33 | best_pre= 96.43 | best_post= 95.37\n",
      "Client  34 | best_pre= 81.87 | best_post= 74.37\n",
      "Client  35 | best_pre= 82.47 | best_post= 70.67\n",
      "Client  36 | best_pre= 98.73 | best_post= 98.40\n",
      "Client  37 | best_pre= 99.20 | best_post= 99.00\n",
      "Client  38 | best_pre= 95.97 | best_post= 94.27\n",
      "Client  39 | best_pre= 81.70 | best_post= 81.57\n",
      "Client  40 | best_pre= 99.57 | best_post= 99.53\n",
      "Client  41 | best_pre= 99.10 | best_post= 98.13\n",
      "Client  42 | best_pre= 82.47 | best_post= 76.10\n",
      "Client  43 | best_pre= 99.20 | best_post= 98.43\n",
      "Client  44 | best_pre= 96.57 | best_post= 95.33\n",
      "Client  45 | best_pre= 98.73 | best_post= 96.57\n",
      "Client  46 | best_pre= 98.73 | best_post= 98.67\n",
      "Client  47 | best_pre= 82.93 | best_post= 81.30\n",
      "Client  48 | best_pre= 96.23 | best_post= 95.83\n",
      "Client  49 | best_pre= 99.60 | best_post= 99.60\n",
      "Client  50 | best_pre= 95.87 | best_post= 92.07\n",
      "Client  51 | best_pre= 99.53 | best_post= 99.50\n",
      "Client  52 | best_pre= 99.60 | best_post= 99.37\n",
      "Client  53 | best_pre= 99.20 | best_post= 98.87\n",
      "Client  54 | best_pre= 99.57 | best_post= 99.57\n",
      "Client  55 | best_pre= 99.60 | best_post= 99.50\n",
      "Client  56 | best_pre= 96.37 | best_post= 95.10\n",
      "Client  57 | best_pre= 72.73 | best_post= 66.37\n",
      "Client  58 | best_pre= 82.47 | best_post= 75.13\n",
      "Client  59 | best_pre= 99.17 | best_post= 99.03\n",
      "Client  60 | best_pre= 99.60 | best_post= 99.43\n",
      "Client  61 | best_pre= 81.57 | best_post= 81.47\n",
      "Client  62 | best_pre= 98.73 | best_post= 98.47\n",
      "Client  63 | best_pre= 99.03 | best_post= 99.23\n",
      "Client  64 | best_pre= 96.00 | best_post= 93.87\n",
      "Client  65 | best_pre= 98.73 | best_post= 98.90\n",
      "Client  66 | best_pre= 99.60 | best_post= 99.47\n",
      "Client  67 | best_pre= 81.87 | best_post= 77.93\n",
      "Client  68 | best_pre= 98.73 | best_post= 96.87\n",
      "Client  69 | best_pre= 99.03 | best_post= 99.13\n",
      "Client  70 | best_pre= 99.60 | best_post= 99.57\n",
      "Client  71 | best_pre= 99.60 | best_post= 99.53\n",
      "Client  72 | best_pre= 98.63 | best_post= 98.00\n",
      "Client  73 | best_pre= 99.03 | best_post= 99.03\n",
      "Client  74 | best_pre= 81.87 | best_post= 47.63\n",
      "Client  75 | best_pre= 82.93 | best_post= 76.93\n",
      "Client  76 | best_pre= 96.57 | best_post= 96.03\n",
      "Client  77 | best_pre= 81.70 | best_post= 60.70\n",
      "Client  78 | best_pre= 81.70 | best_post= 69.03\n",
      "Client  79 | best_pre= 98.73 | best_post= 98.63\n",
      "Client  80 | best_pre= 98.73 | best_post= 98.80\n",
      "Client  81 | best_pre= 96.00 | best_post= 95.27\n",
      "Client  82 | best_pre= 99.60 | best_post= 99.53\n",
      "Client  83 | best_pre= 99.10 | best_post= 98.07\n",
      "Client  84 | best_pre= 98.70 | best_post= 98.57\n",
      "Client  85 | best_pre= 98.67 | best_post= 98.63\n",
      "Client  86 | best_pre= 98.87 | best_post= 96.27\n",
      "Client  87 | best_pre= 99.10 | best_post= 98.90\n",
      "Client  88 | best_pre= 96.50 | best_post= 94.57\n",
      "Client  89 | best_pre= 81.70 | best_post= 78.10\n",
      "Client  90 | best_pre= 81.57 | best_post= 79.40\n",
      "Client  91 | best_pre= 96.70 | best_post= 95.90\n",
      "Client  92 | best_pre= 96.17 | best_post= 95.40\n",
      "Client  93 | best_pre= 96.57 | best_post= 95.23\n",
      "Client  94 | best_pre= 98.73 | best_post= 98.27\n",
      "Client  95 | best_pre= 98.67 | best_post= 98.33\n",
      "Client  96 | best_pre= 81.70 | best_post= 78.80\n",
      "Client  97 | best_pre= 99.20 | best_post= 98.33\n",
      "Client  98 | best_pre= 98.67 | best_post= 98.10\n",
      "Client  99 | best_pre= 98.73 | best_post= 98.63\n",
      "\n",
      "##### ROUND 32 #####\n",
      "\n",
      "##### ROUND 33 #####\n",
      "\n",
      "##### ROUND 34 #####\n",
      "\n",
      "##### ROUND 35 #####\n",
      "\n",
      "##### ROUND 36 #####\n",
      "\n",
      "##### ROUND 37 #####\n",
      "\n",
      "##### ROUND 38 #####\n",
      "\n",
      "##### ROUND 39 #####\n",
      "\n",
      "##### ROUND 40 #####\n",
      "\n",
      "##### ROUND 41 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.0146\n",
      "avg best  : 95.20%\n",
      "\n",
      "Client   0 | best_pre= 99.20 | best_post= 98.97\n",
      "Client   1 | best_pre= 99.17 | best_post= 98.40\n",
      "Client   2 | best_pre= 96.77 | best_post= 96.70\n",
      "Client   3 | best_pre= 98.77 | best_post= 98.80\n",
      "Client   4 | best_pre= 98.97 | best_post= 99.03\n",
      "Client   5 | best_pre= 98.80 | best_post= 98.77\n",
      "Client   6 | best_pre= 96.60 | best_post= 96.17\n",
      "Client   7 | best_pre= 99.20 | best_post= 98.67\n",
      "Client   8 | best_pre= 99.57 | best_post= 99.57\n",
      "Client   9 | best_pre= 99.27 | best_post= 99.20\n",
      "Client  10 | best_pre= 96.70 | best_post= 96.77\n",
      "Client  11 | best_pre= 98.80 | best_post= 98.83\n",
      "Client  12 | best_pre= 98.80 | best_post= 97.70\n",
      "Client  13 | best_pre= 96.53 | best_post= 94.87\n",
      "Client  14 | best_pre= 82.47 | best_post= 80.40\n",
      "Client  15 | best_pre= 82.93 | best_post= 80.63\n",
      "Client  16 | best_pre= 96.27 | best_post= 94.27\n",
      "Client  17 | best_pre= 99.10 | best_post= 98.20\n",
      "Client  18 | best_pre= 96.50 | best_post= 96.47\n",
      "Client  19 | best_pre= 99.03 | best_post= 99.07\n",
      "Client  20 | best_pre= 99.60 | best_post= 99.23\n",
      "Client  21 | best_pre= 82.47 | best_post= 70.73\n",
      "Client  22 | best_pre= 99.27 | best_post= 98.97\n",
      "Client  23 | best_pre= 96.60 | best_post= 94.40\n",
      "Client  24 | best_pre= 96.57 | best_post= 96.00\n",
      "Client  25 | best_pre= 96.70 | best_post= 96.37\n",
      "Client  26 | best_pre= 83.50 | best_post= 78.80\n",
      "Client  27 | best_pre= 98.80 | best_post= 98.33\n",
      "Client  28 | best_pre= 96.60 | best_post= 93.97\n",
      "Client  29 | best_pre= 99.30 | best_post= 99.30\n",
      "Client  30 | best_pre= 99.60 | best_post= 99.57\n",
      "Client  31 | best_pre= 99.60 | best_post= 99.60\n",
      "Client  32 | best_pre= 96.60 | best_post= 90.20\n",
      "Client  33 | best_pre= 96.43 | best_post= 95.37\n",
      "Client  34 | best_pre= 81.87 | best_post= 74.37\n",
      "Client  35 | best_pre= 82.47 | best_post= 70.67\n",
      "Client  36 | best_pre= 98.77 | best_post= 98.53\n",
      "Client  37 | best_pre= 99.20 | best_post= 99.00\n",
      "Client  38 | best_pre= 96.53 | best_post= 95.87\n",
      "Client  39 | best_pre= 81.70 | best_post= 81.57\n",
      "Client  40 | best_pre= 99.57 | best_post= 99.53\n",
      "Client  41 | best_pre= 99.27 | best_post= 98.13\n",
      "Client  42 | best_pre= 82.47 | best_post= 76.10\n",
      "Client  43 | best_pre= 99.27 | best_post= 98.67\n",
      "Client  44 | best_pre= 96.57 | best_post= 95.33\n",
      "Client  45 | best_pre= 98.80 | best_post= 97.13\n",
      "Client  46 | best_pre= 98.77 | best_post= 98.80\n",
      "Client  47 | best_pre= 82.93 | best_post= 81.30\n",
      "Client  48 | best_pre= 96.77 | best_post= 96.20\n",
      "Client  49 | best_pre= 99.60 | best_post= 99.60\n",
      "Client  50 | best_pre= 96.77 | best_post= 93.00\n",
      "Client  51 | best_pre= 99.57 | best_post= 99.50\n",
      "Client  52 | best_pre= 99.60 | best_post= 99.37\n",
      "Client  53 | best_pre= 99.30 | best_post= 98.93\n",
      "Client  54 | best_pre= 99.57 | best_post= 99.57\n",
      "Client  55 | best_pre= 99.60 | best_post= 99.57\n",
      "Client  56 | best_pre= 96.37 | best_post= 95.27\n",
      "Client  57 | best_pre= 83.50 | best_post= 73.83\n",
      "Client  58 | best_pre= 83.50 | best_post= 75.53\n",
      "Client  59 | best_pre= 99.30 | best_post= 99.27\n",
      "Client  60 | best_pre= 99.60 | best_post= 99.43\n",
      "Client  61 | best_pre= 82.10 | best_post= 82.13\n",
      "Client  62 | best_pre= 98.77 | best_post= 98.63\n",
      "Client  63 | best_pre= 99.03 | best_post= 99.23\n",
      "Client  64 | best_pre= 96.00 | best_post= 93.87\n",
      "Client  65 | best_pre= 99.20 | best_post= 99.20\n",
      "Client  66 | best_pre= 99.60 | best_post= 99.47\n",
      "Client  67 | best_pre= 82.10 | best_post= 79.10\n",
      "Client  68 | best_pre= 98.77 | best_post= 97.53\n",
      "Client  69 | best_pre= 99.27 | best_post= 99.27\n",
      "Client  70 | best_pre= 99.60 | best_post= 99.57\n",
      "Client  71 | best_pre= 99.60 | best_post= 99.53\n",
      "Client  72 | best_pre= 98.77 | best_post= 98.67\n",
      "Client  73 | best_pre= 99.20 | best_post= 99.20\n",
      "Client  74 | best_pre= 81.87 | best_post= 48.83\n",
      "Client  75 | best_pre= 82.93 | best_post= 76.93\n",
      "Client  76 | best_pre= 96.57 | best_post= 96.20\n",
      "Client  77 | best_pre= 81.70 | best_post= 61.97\n",
      "Client  78 | best_pre= 81.70 | best_post= 69.03\n",
      "Client  79 | best_pre= 98.80 | best_post= 98.77\n",
      "Client  80 | best_pre= 98.73 | best_post= 98.80\n",
      "Client  81 | best_pre= 96.00 | best_post= 95.50\n",
      "Client  82 | best_pre= 99.60 | best_post= 99.53\n",
      "Client  83 | best_pre= 99.30 | best_post= 98.07\n",
      "Client  84 | best_pre= 98.80 | best_post= 98.63\n",
      "Client  85 | best_pre= 98.80 | best_post= 98.87\n",
      "Client  86 | best_pre= 99.20 | best_post= 96.90\n",
      "Client  87 | best_pre= 99.30 | best_post= 98.93\n",
      "Client  88 | best_pre= 96.77 | best_post= 94.97\n",
      "Client  89 | best_pre= 81.93 | best_post= 78.33\n",
      "Client  90 | best_pre= 82.07 | best_post= 80.57\n",
      "Client  91 | best_pre= 96.77 | best_post= 96.30\n",
      "Client  92 | best_pre= 96.77 | best_post= 96.03\n",
      "Client  93 | best_pre= 96.73 | best_post= 96.07\n",
      "Client  94 | best_pre= 98.80 | best_post= 98.30\n",
      "Client  95 | best_pre= 98.77 | best_post= 98.40\n",
      "Client  96 | best_pre= 82.10 | best_post= 80.77\n",
      "Client  97 | best_pre= 99.30 | best_post= 98.33\n",
      "Client  98 | best_pre= 98.77 | best_post= 98.10\n",
      "Client  99 | best_pre= 98.77 | best_post= 98.70\n",
      "\n",
      "##### ROUND 42 #####\n",
      "\n",
      "##### ROUND 43 #####\n",
      "\n",
      "##### ROUND 44 #####\n",
      "\n",
      "##### ROUND 45 #####\n",
      "\n",
      "##### ROUND 46 #####\n",
      "\n",
      "##### ROUND 47 #####\n",
      "\n",
      "##### ROUND 48 #####\n",
      "\n",
      "##### ROUND 49 #####\n",
      "\n",
      "##### ROUND 50 #####\n"
     ]
    }
   ],
   "source": [
    "#Dual-encoder intiation and FL training\n",
    "\n",
    "\n",
    "\n",
    "# ============================================================\n",
    "# Helpers\n",
    "# ============================================================\n",
    "\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",
    "def fedavg_delta(delta_list, weights):\n",
    "    \"\"\"\n",
    "    FedAvg for *additive updates*  Δθ.\n",
    "    Keys are already identical across dicts.\n",
    "    \"\"\"\n",
    "    tot = float(sum(weights))\n",
    "    out = copy.deepcopy(delta_list[0])\n",
    "    with torch.no_grad():\n",
    "        for k in out:\n",
    "            out[k].zero_()\n",
    "            for w, d in zip(weights, delta_list):\n",
    "                out[k] += (w / tot) * d[k]\n",
    "    return out\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "def fedavg_encoder(encoder_list: list[nn.Module]) -> nn.Module:\n",
    "    \"\"\"\n",
    "    Return a copy of encoder_list[0] whose parameters are the\n",
    "    element-wise average of all encoders in the list.\n",
    "    \"\"\"\n",
    "    n = len(encoder_list)\n",
    "    assert n > 0, \"encoder_list must be non-empty\"\n",
    "\n",
    "    # clone first encoder as template\n",
    "    avg_enc = copy.deepcopy(encoder_list[0])\n",
    "    avg_sd  = avg_enc.state_dict()           # OrderedDict of tensors\n",
    "\n",
    "    with torch.no_grad():\n",
    "        for k in avg_sd.keys():\n",
    "            stacked = torch.stack([enc.state_dict()[k] for enc in encoder_list], dim=0)\n",
    "            avg_sd[k].copy_(stacked.mean(dim=0))\n",
    "\n",
    "    avg_enc.load_state_dict(avg_sd)\n",
    "    return avg_enc\n",
    "\n",
    "\n",
    "def make_combined_model_from_single(\n",
    "    single_model: nn.Module,\n",
    "    cluster_enc: nn.Module,\n",
    "    num_classes: int = 10,\n",
    "    dataset: str = \"mnist\"\n",
    ") -> nn.Module:\n",
    "    \"\"\"\n",
    "    single_model : instance of SimpleCNNMNIST2 or SimpleCNN\n",
    "    cluster_enc  : averaged encoder to use as PRIMARY encoder\n",
    "    dataset      : string, e.g., 'mnist', 'fmnist', 'svhn', 'cifar10', 'cifar100'\n",
    "    \"\"\"\n",
    "    # fresh secondary encoder (same dim) – random init\n",
    "    #sec_enc = clone_encoder(single_model.encoder)\n",
    "    sec_enc = copy.deepcopy(cluster_enc)  # same weights as primary\n",
    "\n",
    "    # input dim to classifier\n",
    "    in_dim = 84 if dataset in {\"mnist\", \"fmnist\"} else 256\n",
    "    clf_input_dim = 2 * in_dim\n",
    "    new_clf = nn.Linear(clf_input_dim, num_classes)\n",
    "    clf_sd = new_clf.state_dict()\n",
    "\n",
    "    if dataset in {\"mnist\", \"fmnist\"}:\n",
    "        return CombinedModelMNIST(cluster_enc, sec_enc, clf_sd, num_classes)\n",
    "    elif dataset in {\"svhn\", \"cifar10\", \"cifar100\"}:\n",
    "        return CombinedSimpleCNN(cluster_enc, sec_enc, clf_sd, num_classes)\n",
    "    else:\n",
    "        raise ValueError(f\"Unsupported dataset: {dataset}\")\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",
    "    dataset=\"mnist\"    # <<< pass the dataset name here\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",
    "    dataset             : str – 'mnist', 'fmnist', 'svhn', 'cifar10', 'cifar100'\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",
    "        dataset=dataset\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",
    "w_glob_per_cluster = [None] * len(clusters)\n",
    "\n",
    "\n",
    "w_glob_per_cluster = [None] * len(clusters)\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",
    "        dataset=args.dataset   # <<< set this based on your current run\n",
    "    )\n",
    "print(\">>> All clusters switched to combined \")\n",
    "\n",
    "\n",
    "\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",
    "# Helpers\n",
    "# ============================================================\n",
    "\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",
    "}\n",
    "\n",
    "# ============================================================\n",
    "# Clustered-FL primary-update loop\n",
    "# ============================================================\n",
    "#  ADD THIS near the other imports in client_cluster_fl.py\n",
    "\n",
    "n_rounds   = 201\n",
    "print_step = 10              # stats cadence after warm-up\n",
    "loss_buf   = []\n",
    "\n",
    "for rnd in range(n_rounds):\n",
    "\n",
    "    print(f\"\\n##### ROUND {rnd+1} #####\")\n",
    "\n",
    "    # 1) sample users\n",
    "    m            = int(args.frac * args.num_users)\n",
    "    idxs_users   = np.random.choice(args.num_users, m, replace=False)\n",
    "    clusters_now = cluster_participants(idxs_users, clients_clust_id)\n",
    "\n",
    "    # 2) broadcast combined cluster model to each selected user\n",
    "    for uid in idxs_users:\n",
    "        cid = clients_clust_id[uid]\n",
    "        clients[uid].set_state_dict(copy.deepcopy(w_glob_per_cluster[cid]))\n",
    "\n",
    "    # 3) per-cluster FedAvg containers\n",
    "    enc_pool, clf_pool, freq_pool = {}, {}, {}\n",
    "\n",
    "    # 4) local training\n",
    "    for uid in idxs_users:\n",
    "        cid   = clients_clust_id[uid]\n",
    "        ndata = len(net_dataidx_map[uid])\n",
    "\n",
    "        # ---- metrics before train\n",
    "        pre_loss, pre_acc = clients[uid].eval_test()\n",
    "        client_metric[uid][\"best_pre\"]  = max(client_metric[uid][\"best_pre\"],  pre_acc)\n",
    "\n",
    "        # ---- local SGD (own_encoder + classifier only)\n",
    "        loss = clients[uid].train2()\n",
    "        loss_buf.append(loss)\n",
    "\n",
    "        # ---- metrics after train\n",
    "        post_loss, post_acc = clients[uid].eval_test()\n",
    "        client_metric[uid][\"best_post\"] = max(client_metric[uid][\"best_post\"], post_acc)\n",
    "        client_metric[uid][\"best_any\"]  = max(client_metric[uid][\"best_any\"],\n",
    "                                              pre_acc, post_acc)\n",
    "\n",
    "        # ---- stash trainable weights for FedAvg\n",
    "        # Wrap encoder weights with prefix\n",
    "        own_sd = clients[uid].net.own_encoder.state_dict()\n",
    "        own_sd_prefixed = OrderedDict({f\"own_encoder.{k}\": v for k, v in own_sd.items()})\n",
    "        enc_pool.setdefault(cid, []).append(own_sd_prefixed)\n",
    "        \n",
    "        # Wrap classifier weights with prefix\n",
    "        clf_sd = clients[uid].net.classifier.state_dict()\n",
    "        clf_sd_prefixed = OrderedDict({f\"classifier.{k}\": v for k, v in clf_sd.items()})\n",
    "        clf_pool.setdefault(cid, []).append(clf_sd_prefixed)\n",
    "\n",
    "    # 5) FedAvg per cluster (trainable blocks)\n",
    "    for cid in enc_pool:\n",
    "        weights = make_weight_vec(clusters_now[cid], net_dataidx_map)\n",
    "        upd_enc = FedAvg_trainable(enc_pool[cid], weights)\n",
    "        upd_clf = FedAvg_trainable(clf_pool[cid], weights)\n",
    "        w_glob_per_cluster[cid].update(upd_enc)\n",
    "        w_glob_per_cluster[cid].update(upd_clf)\n",
    "    \n",
    "    # ------------------------------------------------------------------\n",
    "    # 6)  Secondary-encoder enrichment  (ONLY for sampled clusters)\n",
    "\n",
    "    local_sec_ep = 10\n",
    "    sec_lr       = 0.01\n",
    "    \n",
    "    for z, clist_sampled in clusters_now.items():\n",
    "    \n",
    "        learners = H_out.get(z, [])\n",
    "        if not learners or len(clist_sampled) == 0:\n",
    "            continue\n",
    "    \n",
    "        # 6.1  Θ2′z  (average current learner weights)\n",
    "        learner_secs = [\n",
    "            {k: v.clone() for k, v in w_glob_per_cluster[j].items()\n",
    "             if k.startswith('secondary_encoder.')}\n",
    "            for j in learners\n",
    "        ]\n",
    "        θ2_prime = fedavg_secondary(learner_secs, [1] * len(learner_secs))\n",
    "    \n",
    "        # strip prefix for bare encoder\n",
    "        base_sec_state = OrderedDict(\n",
    "            (k.replace(\"secondary_encoder.\", \"\"), v) for k, v in θ2_prime.items()\n",
    "        )\n",
    "    \n",
    "        # 6.3  each sampled client returns Δθ_i\n",
    "        delta_list, sizes = [], []\n",
    "        for uid in clist_sampled:\n",
    "            Δθ = clients[uid].train_secondary(base_sec_state,\n",
    "                                              epochs=local_sec_ep,\n",
    "                                              lr=sec_lr)\n",
    "            delta_list.append(Δθ)\n",
    "            sizes.append(len(net_dataidx_map[uid]))\n",
    "    \n",
    "        # 6.4  FedAvg over Δθ_i   →   Δ̄θ(z)\n",
    "        Δ̄θ = fedavg_delta(delta_list, sizes)\n",
    "    \n",
    "        # 6.5  apply Δ̄θ(z) to every learner cluster j\n",
    "        for j in learners:\n",
    "            for k, dv in Δ̄θ.items():                         # k = \"0.weight\", …\n",
    "                key = f\"secondary_encoder.{k}\"               # add prefix\n",
    "                w_glob_per_cluster[j][key] += dv.clone()     # IN-PLACE ADD\n",
    "\n",
    "\n",
    "    # 7) periodic statistics\n",
    "    if rnd < 4:\n",
    "        print_step = 1\n",
    "    else:\n",
    "        print_step = 10\n",
    "\n",
    "    if rnd % print_step == 0:\n",
    "        avg_loss = np.mean(loss_buf) if loss_buf else 0.0\n",
    "        avg_pre  = np.mean([m[\"best_pre\"]  for m in client_metric.values()])\n",
    "        avg_post = np.mean([m[\"best_post\"] for m in client_metric.values()])\n",
    "        avg_any  = np.mean([m[\"best_any\"]  for m in client_metric.values()])\n",
    "\n",
    "        print(\"---- ROUND STATS ----\")\n",
    "        print(f\"avg train loss : {avg_loss:.4f}\")\n",
    "        print(f\"avg best  : {avg_any :.2f}%\\n\")\n",
    "\n",
    "        # optional per-client line (comment if too verbose)\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",
    "\n",
    "        # reset buffers for next report window\n",
    "        loss_buf.clear()\n",
    "        gc.collect()"
   ]
  },
  {
   "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
}
