{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Paramters :  Namespace(rounds=81, num_users=100, nclass=2, nsample_pc=250, frac=0.2, local_ep=10, local_bs=10, bs=128, lr=0.01, momentum=0.5, warmup_epoch=0, trial=1, mu=0.001, model='simple-cnn', ks=5, in_ch=3, dataset='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": 2,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "partition: flag-non-iid\n",
      "Data statistics Train:\n",
      " {0: {2: 16, 6: 102, 7: 75}, 1: {3: 344, 8: 348, 9: 705}, 2: {2: 118, 4: 195, 6: 472}, 3: {2: 585, 4: 85, 6: 40}, 4: {3: 1834, 8: 108, 9: 402}, 5: {0: 249, 4: 30, 5: 180}, 6: {0: 184, 1: 712, 5: 75}, 7: {3: 310, 8: 276, 9: 117}, 8: {2: 323, 6: 26, 7: 873}, 9: {0: 79, 1: 6, 5: 8}, 10: {3: 75, 8: 202, 9: 152}, 11: {2: 22, 4: 156, 6: 516}, 12: {2: 311, 4: 388, 6: 17}, 13: {3: 238, 8: 417, 9: 251}, 14: {2: 176, 4: 36, 6: 136}, 15: {0: 169, 1: 68, 5: 151}, 16: {2: 13, 4: 404, 6: 303}, 17: {0: 324, 1: 381, 5: 62}, 18: {2: 208, 6: 19, 7: 38}, 19: {0: 44, 1: 139, 5: 260}, 20: {0: 173, 4: 231, 5: 116}, 21: {2: 28, 6: 21, 7: 260}, 22: {0: 77, 4: 265, 5: 234}, 23: {2: 144, 6: 34, 7: 534}, 24: {0: 21, 1: 314, 5: 268}, 25: {0: 274, 4: 206, 5: 94}, 26: {0: 192, 1: 168, 5: 110}, 27: {3: 20, 8: 391, 9: 512}, 28: {0: 2, 4: 302, 5: 11}, 29: {0: 118, 4: 134, 5: 153}, 30: {2: 135, 6: 6, 7: 55}, 31: {2: 47, 4: 39, 6: 194}, 32: {0: 144, 1: 39, 5: 27}, 33: {0: 91, 1: 48, 5: 129}, 34: {2: 120, 4: 51, 6: 433}, 35: {3: 300, 8: 332, 9: 442}, 36: {3: 98, 8: 202, 9: 586}, 37: {2: 334, 4: 493, 6: 305}, 38: {0: 42, 1: 103, 5: 595}, 39: {2: 134, 6: 11, 7: 269}, 40: {2: 191, 6: 114, 7: 15}, 41: {0: 19, 1: 14, 5: 99}, 42: {0: 209, 1: 616, 5: 158}, 43: {0: 97, 4: 104, 5: 155}, 44: {0: 158, 4: 287, 5: 268}, 45: {0: 269, 4: 202, 5: 12}, 46: {3: 108, 8: 766, 9: 366}, 47: {3: 5, 8: 280, 9: 37}, 48: {0: 24, 1: 150, 5: 206}, 49: {0: 115, 4: 171, 5: 100}, 50: {2: 10, 6: 87, 7: 22}, 51: {2: 74, 6: 11, 7: 531}, 52: {0: 185, 4: 225, 5: 398}, 53: {0: 90, 4: 79, 5: 157}, 54: {0: 193, 4: 237, 5: 121}, 55: {0: 414, 1: 61, 5: 77}, 56: {2: 75, 4: 68, 6: 227}, 57: {3: 75, 8: 490, 9: 254}, 58: {2: 409, 6: 138, 7: 124}, 59: {3: 630, 8: 10, 9: 377}, 60: {0: 200, 1: 214, 5: 56}, 61: {3: 207, 8: 595, 9: 59}, 62: {2: 136, 6: 55, 7: 300}, 63: {0: 393, 1: 595, 5: 63}, 64: {2: 91, 6: 3, 7: 7}, 65: {2: 17, 6: 39, 7: 255}, 66: {0: 188, 4: 10, 5: 171}, 67: {2: 12, 4: 817, 6: 131}, 68: {3: 277, 8: 63, 9: 375}, 69: {3: 382, 8: 542, 9: 166}, 70: {2: 120, 4: 114, 6: 239}, 71: {0: 100, 4: 104, 5: 173}, 72: {0: 17, 1: 1, 5: 39}, 73: {3: 20, 8: 450, 9: 734}, 74: {0: 82, 1: 1251, 5: 57}, 75: {2: 23, 6: 32, 7: 536}, 76: {2: 9, 6: 152, 7: 150}, 77: {2: 27, 6: 54, 7: 146}, 78: {0: 36, 1: 323, 5: 218}, 79: {2: 255, 4: 70, 6: 165}, 80: {0: 115, 4: 4, 5: 17}, 81: {0: 120, 4: 2, 5: 28}, 82: {0: 201, 1: 110, 5: 48}, 83: {2: 81, 6: 356, 7: 554}, 84: {3: 896, 8: 91, 9: 361}, 85: {2: 769, 4: 28, 6: 202}, 86: {2: 21, 4: 28, 6: 4}, 87: {2: 201, 6: 871, 7: 134}, 88: {2: 219, 6: 156, 7: 188}, 89: {0: 83, 1: 101, 5: 215}, 90: {2: 117, 6: 132, 7: 273}, 91: {3: 181, 8: 437, 9: 104}, 92: {2: 127, 6: 39, 7: 661}, 93: {0: 32, 4: 66, 5: 415}, 94: {0: 198, 4: 118, 5: 104}, 95: {0: 143, 1: 586, 5: 93}, 96: {2: 155, 4: 111, 6: 124}, 97: {2: 109, 4: 16, 6: 25}, 98: {0: 136, 4: 84, 5: 79}, 99: {2: 38, 4: 40, 6: 9}} \n",
      "\n",
      "Data statistics Test:\n",
      " {0: {2: 1000, 6: 1000, 7: 1000}, 1: {3: 1000, 8: 1000, 9: 1000}, 2: {2: 1000, 4: 1000, 6: 1000}, 3: {2: 1000, 4: 1000, 6: 1000}, 4: {3: 1000, 8: 1000, 9: 1000}, 5: {0: 1000, 4: 1000, 5: 1000}, 6: {0: 1000, 1: 1000, 5: 1000}, 7: {3: 1000, 8: 1000, 9: 1000}, 8: {2: 1000, 6: 1000, 7: 1000}, 9: {0: 1000, 1: 1000, 5: 1000}, 10: {3: 1000, 8: 1000, 9: 1000}, 11: {2: 1000, 4: 1000, 6: 1000}, 12: {2: 1000, 4: 1000, 6: 1000}, 13: {3: 1000, 8: 1000, 9: 1000}, 14: {2: 1000, 4: 1000, 6: 1000}, 15: {0: 1000, 1: 1000, 5: 1000}, 16: {2: 1000, 4: 1000, 6: 1000}, 17: {0: 1000, 1: 1000, 5: 1000}, 18: {2: 1000, 6: 1000, 7: 1000}, 19: {0: 1000, 1: 1000, 5: 1000}, 20: {0: 1000, 4: 1000, 5: 1000}, 21: {2: 1000, 6: 1000, 7: 1000}, 22: {0: 1000, 4: 1000, 5: 1000}, 23: {2: 1000, 6: 1000, 7: 1000}, 24: {0: 1000, 1: 1000, 5: 1000}, 25: {0: 1000, 4: 1000, 5: 1000}, 26: {0: 1000, 1: 1000, 5: 1000}, 27: {3: 1000, 8: 1000, 9: 1000}, 28: {0: 1000, 4: 1000, 5: 1000}, 29: {0: 1000, 4: 1000, 5: 1000}, 30: {2: 1000, 6: 1000, 7: 1000}, 31: {2: 1000, 4: 1000, 6: 1000}, 32: {0: 1000, 1: 1000, 5: 1000}, 33: {0: 1000, 1: 1000, 5: 1000}, 34: {2: 1000, 4: 1000, 6: 1000}, 35: {3: 1000, 8: 1000, 9: 1000}, 36: {3: 1000, 8: 1000, 9: 1000}, 37: {2: 1000, 4: 1000, 6: 1000}, 38: {0: 1000, 1: 1000, 5: 1000}, 39: {2: 1000, 6: 1000, 7: 1000}, 40: {2: 1000, 6: 1000, 7: 1000}, 41: {0: 1000, 1: 1000, 5: 1000}, 42: {0: 1000, 1: 1000, 5: 1000}, 43: {0: 1000, 4: 1000, 5: 1000}, 44: {0: 1000, 4: 1000, 5: 1000}, 45: {0: 1000, 4: 1000, 5: 1000}, 46: {3: 1000, 8: 1000, 9: 1000}, 47: {3: 1000, 8: 1000, 9: 1000}, 48: {0: 1000, 1: 1000, 5: 1000}, 49: {0: 1000, 4: 1000, 5: 1000}, 50: {2: 1000, 6: 1000, 7: 1000}, 51: {2: 1000, 6: 1000, 7: 1000}, 52: {0: 1000, 4: 1000, 5: 1000}, 53: {0: 1000, 4: 1000, 5: 1000}, 54: {0: 1000, 4: 1000, 5: 1000}, 55: {0: 1000, 1: 1000, 5: 1000}, 56: {2: 1000, 4: 1000, 6: 1000}, 57: {3: 1000, 8: 1000, 9: 1000}, 58: {2: 1000, 6: 1000, 7: 1000}, 59: {3: 1000, 8: 1000, 9: 1000}, 60: {0: 1000, 1: 1000, 5: 1000}, 61: {3: 1000, 8: 1000, 9: 1000}, 62: {2: 1000, 6: 1000, 7: 1000}, 63: {0: 1000, 1: 1000, 5: 1000}, 64: {2: 1000, 6: 1000, 7: 1000}, 65: {2: 1000, 6: 1000, 7: 1000}, 66: {0: 1000, 4: 1000, 5: 1000}, 67: {2: 1000, 4: 1000, 6: 1000}, 68: {3: 1000, 8: 1000, 9: 1000}, 69: {3: 1000, 8: 1000, 9: 1000}, 70: {2: 1000, 4: 1000, 6: 1000}, 71: {0: 1000, 4: 1000, 5: 1000}, 72: {0: 1000, 1: 1000, 5: 1000}, 73: {3: 1000, 8: 1000, 9: 1000}, 74: {0: 1000, 1: 1000, 5: 1000}, 75: {2: 1000, 6: 1000, 7: 1000}, 76: {2: 1000, 6: 1000, 7: 1000}, 77: {2: 1000, 6: 1000, 7: 1000}, 78: {0: 1000, 1: 1000, 5: 1000}, 79: {2: 1000, 4: 1000, 6: 1000}, 80: {0: 1000, 4: 1000, 5: 1000}, 81: {0: 1000, 4: 1000, 5: 1000}, 82: {0: 1000, 1: 1000, 5: 1000}, 83: {2: 1000, 6: 1000, 7: 1000}, 84: {3: 1000, 8: 1000, 9: 1000}, 85: {2: 1000, 4: 1000, 6: 1000}, 86: {2: 1000, 4: 1000, 6: 1000}, 87: {2: 1000, 6: 1000, 7: 1000}, 88: {2: 1000, 6: 1000, 7: 1000}, 89: {0: 1000, 1: 1000, 5: 1000}, 90: {2: 1000, 6: 1000, 7: 1000}, 91: {3: 1000, 8: 1000, 9: 1000}, 92: {2: 1000, 6: 1000, 7: 1000}, 93: {0: 1000, 4: 1000, 5: 1000}, 94: {0: 1000, 4: 1000, 5: 1000}, 95: {0: 1000, 1: 1000, 5: 1000}, 96: {2: 1000, 4: 1000, 6: 1000}, 97: {2: 1000, 4: 1000, 6: 1000}, 98: {0: 1000, 4: 1000, 5: 1000}, 99: {2: 1000, 4: 1000, 6: 1000}} \n",
      "\n"
     ]
    }
   ],
   "source": [
    "##################################### Data partitioning section \n",
    "args.local_view = True\n",
    "X_train, y_train, X_test, y_test, net_dataidx_map, net_dataidx_map_test, \\\n",
    "traindata_cls_counts, testdata_cls_counts = partition_data(args.dataset, \n",
    "args.datadir, args.logdir, args.partition, args.num_users, beta=args.beta, local_view=args.local_view)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "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: [2 6 7], Counts: [ 16 102  75]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [344 348 705]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [118 195 472]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [585  85  40]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [1834  108  402]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [249  30 180]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [184 712  75]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [310 276 117]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [323  26 873]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [79  6  8]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [ 75 202 152]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [ 22 156 516]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [311 388  17]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [238 417 251]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [176  36 136]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [169  68 151]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [ 13 404 303]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [324 381  62]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [208  19  38]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [ 44 139 260]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [173 231 116]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [ 28  21 260]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [ 77 265 234]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [144  34 534]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [ 21 314 268]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [274 206  94]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [192 168 110]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [ 20 391 512]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [  2 302  11]\n",
      "Shape of U: (784, 8)\n",
      "Labels: [0 4 5], Counts: [118 134 153]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [135   6  55]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [ 47  39 194]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [144  39  27]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [ 91  48 129]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [120  51 433]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [300 332 442]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [ 98 202 586]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [334 493 305]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [ 42 103 595]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [134  11 269]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [191 114  15]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [19 14 99]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [209 616 158]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [ 97 104 155]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [158 287 268]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [269 202  12]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [108 766 366]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [  5 280  37]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [ 24 150 206]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [115 171 100]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [10 87 22]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [ 74  11 531]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [185 225 398]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [ 90  79 157]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [193 237 121]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [414  61  77]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [ 75  68 227]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [ 75 490 254]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [409 138 124]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [630  10 377]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [200 214  56]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [207 595  59]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [136  55 300]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [393 595  63]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [91  3  7]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [ 17  39 255]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [188  10 171]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [ 12 817 131]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [277  63 375]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [382 542 166]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [120 114 239]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [100 104 173]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [17  1 39]\n",
      "Shape of U: (784, 7)\n",
      "Labels: [3 8 9], Counts: [ 20 450 734]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [  82 1251   57]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [ 23  32 536]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [  9 152 150]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [ 27  54 146]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [ 36 323 218]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [255  70 165]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [115   4  17]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [120   2  28]\n",
      "Shape of U: (784, 8)\n",
      "Labels: [0 1 5], Counts: [201 110  48]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [ 81 356 554]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [896  91 361]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [769  28 202]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [21 28  4]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [201 871 134]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [219 156 188]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [ 83 101 215]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [117 132 273]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [3 8 9], Counts: [181 437 104]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 6 7], Counts: [127  39 661]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [ 32  66 415]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [198 118 104]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 1 5], Counts: [143 586  93]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [155 111 124]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [109  16  25]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [0 4 5], Counts: [136  84  79]\n",
      "Shape of U: (784, 9)\n",
      "Labels: [2 4 6], Counts: [38 40  9]\n",
      "Shape of U: (784, 9)\n",
      "###### Gradient data ROUND 1 ######\n",
      "Grad_similarities:\n",
      "[[ 0.0000 84.0677 76.3268 ... 77.8278 79.4515 76.9002]\n",
      " [84.0677  0.0000 89.1369 ... 95.4171 86.1276 93.5248]\n",
      " [76.3268 89.1369  0.0000 ... 69.7325 83.7175 72.8822]\n",
      " ...\n",
      " [77.8278 95.4171 69.7325 ...  0.0000 81.2300 28.4546]\n",
      " [79.4515 86.1276 83.7175 ... 81.2300  0.0000 76.8046]\n",
      " [76.9002 93.5248 72.8822 ... 28.4546 76.8046  0.0000]]\n"
     ]
    }
   ],
   "source": [
    "train_dl_global, test_dl_global, train_ds_global, test_ds_global = get_dataloader(args.dataset,\n",
    "                                                                                   args.datadir,\n",
    "                                                                                   args.batch_size,\n",
    "                                                                                   32)\n",
    "\n",
    "print(\"len train_ds_global:\", len(train_ds_global))\n",
    "print(\"len test_ds_global:\", len(test_ds_global))\n",
    "\n",
    "\n",
    "################################### build  single fe+cl model\n",
    "def init_nets(args, dropout_p=0.5):\n",
    "\n",
    "    users_model = []\n",
    "\n",
    "    for net_i in range(-1, args.num_users):\n",
    "        if args.model == \"simple-cnn\":\n",
    "            if args.dataset in (\"cifar10\", \"cinic10\", \"svhn\"):\n",
    "                #net = CNN_CIFAR.to(args.device) #changed delete\n",
    "                # net = SimpleCNNLight(input_dim=(16 * 5 * 5), hidden_dims=[120, 84], output_dim=10).to(args.device)\n",
    "                net = SimpleCNN(num_classes=10).to(args.device) \n",
    "            elif args.dataset in (\"mnist\", 'femnist', 'fmnist'):\n",
    "                net = SimpleCNNMNIST2().to(args.device)\n",
    "               # net = SimpleCNNMNIST(input_dim=(16 * 4 * 4), hidden_dims=[120, 84], output_dim=10).to(args.device)\n",
    "            elif args.dataset == 'cifar100':\n",
    "                #net = SimpleCNNLight(input_dim=(16 * 5 * 5), hidden_dims=[120, 84], output_dim=100).to(args.device)\n",
    " \n",
    "                net = SimpleCNN_3(input_dim=(16 * 3 * 5 * 5), hidden_dims=[120*3, 84*3], output_dim=100).to(args.device)\n",
    "                #\n",
    "        elif args.model ==\"simple-cnn-3\":\n",
    "            if args.dataset == 'cifar100': \n",
    "                net = SimpleCNN_3(input_dim=(16 * 3 * 5 * 5), hidden_dims=[120*3, 84*3], output_dim=100).to(args.device)\n",
    "            if args.dataset == 'tinyimagenet':\n",
    "                net = SimpleCNNTinyImagenet_3(input_dim=(16 * 3 * 13 * 13), hidden_dims=[120*3, 84*3], \n",
    "                                              output_dim=200).to(args.device)\n",
    "        # elif args.model == \"vgg-9\":\n",
    "        #     if args.dataset in (\"mnist\", 'femnist'):\n",
    "        #         net = ModerateCNNMNIST().to(args.device)\n",
    "        #     elif args.dataset in (\"cifar10\", \"cinic10\", \"svhn\"):\n",
    "        #         # print(\"in moderate cnn\")\n",
    "        #         net = ModerateCNN().to(args.device)\n",
    "        #     elif args.dataset == 'celeba':\n",
    "        #         net = ModerateCNN(output_dim=2).to(args.device)\n",
    "        elif args.model == 'resnet9': \n",
    "            if args.dataset == 'cifar100':\n",
    "                net = SimpleCNN_3(input_dim=(16 * 3 * 5 * 5), hidden_dims=[120*3, 84*3], output_dim=100).to(args.device)\n",
    "                #net = ResNet9(in_channels=3, num_classes=100)\n",
    "            elif args.dataset == 'tinyimagenet': \n",
    "                net = ResNet9(in_channels=3, num_classes=200, dim=512*2*2)\n",
    "        else:\n",
    "            print(\"not supported yet\")\n",
    "            exit(1)\n",
    "        if net_i == -1: #initializing the global model\n",
    "            net_glob = copy.deepcopy(net) \n",
    "           \n",
    "            initial_state_dict = copy.deepcopy(net_glob.state_dict())\n",
    "            server_state_dict = copy.deepcopy(net_glob.state_dict())\n",
    "            if args.load_initial:\n",
    "                initial_state_dict = torch.load(args.load_initial)\n",
    "                server_state_dict = torch.load(args.load_initial)\n",
    "                net_glob.load_state_dict(initial_state_dict)\n",
    " \n",
    "        else:\n",
    "            users_model.append(copy.deepcopy(net))\n",
    "            users_model[net_i].load_state_dict(initial_state_dict)\n",
    "\n",
    "#     model_meta_data = []\n",
    "#     layer_type = []\n",
    "#     for (k, v) in nets[0].state_dict().items():\n",
    "#         model_meta_data.append(v.shape)\n",
    "#         layer_type.append(k)\n",
    "\n",
    "    return users_model, net_glob, initial_state_dict, server_state_dict\n",
    "\n",
    "print(f'MODEL: {args.model}, Dataset: {args.dataset}')\n",
    "\n",
    "users_model, net_glob, initial_state_dict, server_state_dict = init_nets(args, dropout_p=0.5)\n",
    "\n",
    "print(net_glob)\n",
    "\n",
    "total = 0 \n",
    "for name, param in net_glob.named_parameters():\n",
    "    print(name, param.size())\n",
    "    total += np.prod(param.size())\n",
    "    #print(np.array(param.data.cpu().numpy().reshape([-1])))\n",
    "    #print(isinstance(param.data.cpu().numpy(), np.array))\n",
    "print(total)\n",
    "\n",
    "################################# Initializing Clients \n",
    "traindata_cls_ratio = {}\n",
    "pretrain_data1=[] #changed\n",
    "pretrain_data2=[] #changed\n",
    "\n",
    "budget = 50\n",
    "for i in range(args.num_users):\n",
    "    total_sum = sum(list(traindata_cls_counts[i].values()))\n",
    "    base = 1/len(list(traindata_cls_counts[i].values()))\n",
    "    temp_ratio = {}\n",
    "    for k in traindata_cls_counts[i].keys():\n",
    "        ss = traindata_cls_counts[i][k]/total_sum\n",
    "        temp_ratio[k] = (traindata_cls_counts[i][k]/total_sum)\n",
    "        # if ss >= (base + 0.05): \n",
    "        #     temp_ratio[k] = traindata_cls_counts[i][k]  #if any ratio excceds the base, use the \n",
    "            #actual value instead of ratios\n",
    "            \n",
    "    sub_sum = sum(list(temp_ratio.values()))\n",
    "    for k in temp_ratio.keys():\n",
    "        temp_ratio[k] = (temp_ratio[k]/sub_sum)*budget      #normalize the ratios to budget\n",
    "    \n",
    "    round_ratio = round_to(list(temp_ratio.values()), budget) #round ratios so they are int\n",
    "    cnt = 0 \n",
    "    for k in temp_ratio.keys():\n",
    "        temp_ratio[k] = round_ratio[cnt]\n",
    "        cnt+=1\n",
    "        \n",
    "    traindata_cls_ratio[i] = temp_ratio  \n",
    "    \n",
    "clients = []\n",
    "U_clients = []\n",
    "\n",
    "K = args.n_basis\n",
    "#K = 5\n",
    "U_dict={}\n",
    "for idx in range(args.num_users):\n",
    "    \n",
    "    dataidxs = net_dataidx_map[idx]\n",
    "    if net_dataidx_map_test is None:\n",
    "        dataidx_test = None \n",
    "    else:\n",
    "        dataidxs_test = net_dataidx_map_test[idx]\n",
    "\n",
    "    #print(f'Initializing Client {idx}')\n",
    "\n",
    "    noise_level = args.noise\n",
    "    if idx == args.num_users - 1:\n",
    "        noise_level = 0\n",
    "\n",
    "    if args.noise_type == 'space':\n",
    "        train_dl_local, test_dl_local, train_ds_local, test_ds_local, pre_train_dl, validation_dl = get_dataloader(args.dataset, \n",
    "                                                                       args.datadir, args.local_bs, 32, \n",
    "                                                                       dataidxs, noise_level, idx, \n",
    "                                                                       args.num_users-1, \n",
    "                                                                       dataidxs_test=dataidxs_test)\n",
    "    else:\n",
    "        noise_level = args.noise / (args.num_users - 1) * idx\n",
    "        train_dl_local, test_dl_local, train_ds_local, test_ds_local, pre_train_dl, validation_dl = get_dataloader(args.dataset, \n",
    "                                                                       args.datadir, args.local_bs, 32, \n",
    "                                                                       dataidxs, noise_level, \n",
    "                                                                       dataidxs_test=dataidxs_test)\n",
    "    idxs_local = np.arange(len(train_ds_local.data))\n",
    "    labels_local = np.array(train_ds_local.target)\n",
    "    # Sort Labels Train \n",
    "    idxs_labels_local = np.vstack((idxs_local, labels_local))\n",
    "    idxs_labels_local = idxs_labels_local[:, idxs_labels_local[1, :].argsort()]\n",
    "    idxs_local = idxs_labels_local[0, :]\n",
    "    labels_local = idxs_labels_local[1, :]\n",
    "    \n",
    "    uni_labels, cnt_labels = np.unique(labels_local, return_counts=True)\n",
    "    \n",
    "    print(f'Labels: {uni_labels}, Counts: {cnt_labels}')\n",
    "    \n",
    "    nlabels = len(uni_labels)\n",
    "    cnt = 0\n",
    "    U_temp = []\n",
    "    D_temp = {}\n",
    "    for j in range(nlabels):\n",
    "        local_ds1 = train_ds_local.data[idxs_local[cnt:cnt+cnt_labels[j]]]\n",
    "        x=local_ds1.shape[0]\n",
    "        local_ds1_copy = copy.deepcopy(local_ds1)\n",
    "        #print(train_ds_local[idxs_local[cnt]][0])\n",
    "        #show_tensor_image(train_ds_local[idxs_local[cnt]][0])\n",
    "        local_ds1 = local_ds1.reshape(cnt_labels[j], -1)\n",
    "         \n",
    "        local_ds1 = local_ds1.T\n",
    "        if type(train_ds_local.target[idxs_local[cnt:cnt+cnt_labels[j]]]) == torch.Tensor:\n",
    "            label1 = list(set(train_ds_local.target[idxs_local[cnt:cnt+cnt_labels[j]]].numpy()))\n",
    "        else:\n",
    "            label1 = list(set(train_ds_local.target[idxs_local[cnt:cnt+cnt_labels[j]]]))\n",
    "        assert len(label1) == 1\n",
    "        \n",
    "        #print(f'Label {j} : {label1}')\n",
    "        \n",
    "        \n",
    "        #pretrain_data1.extend(select_representative_images(copy.deepcopy(local_ds1_copy),cnt_labels[j], 8, 30, label1[0])) #changed, delete, para=2nd\n",
    "        # if local_ds1_copy.shape[0]>1:\n",
    "        #     pretrain_data2.extend(select_representative_images_with_svd(copy.deepcopy(local_ds1_copy), x, label1[0]))\n",
    "        if args.partition == 'noniid-labeldir': \n",
    "            #print('Dir partition')\n",
    "            if label1 in list(traindata_cls_ratio[idx].keys()): \n",
    "                K = traindata_cls_ratio[idx][label1[0]]\n",
    "            else: \n",
    "                K = min(args.n_basis,x)    #changed, K = args.n_basis\n",
    "        if K > 0:\n",
    "            u1_temp, sh1_temp, vh1_temp = np.linalg.svd(local_ds1, full_matrices=False)\n",
    "            u1_temp=u1_temp/np.linalg.norm(u1_temp, ord=2, axis=0)\n",
    "            U_temp.append(u1_temp[:, 0:K])\n",
    "            D_temp[label1[0]] = u1_temp[:, 0:K] #changed\n",
    "            \n",
    "        cnt+=cnt_labels[j]\n",
    "        \n",
    "    #U_temp = [u1_temp[:, 0:K], u2_temp[:, 0:K]]\n",
    "    U_clients.append(copy.deepcopy(np.hstack(U_temp)))\n",
    "    U_dict[idx]= D_temp\n",
    "    \n",
    "    print(f'Shape of U: {U_clients[-1].shape}')\n",
    "    \n",
    "    clients.append(Client_ClusterFL(idx, copy.deepcopy(users_model[idx]), args.local_bs, args.local_ep, \n",
    "               args.lr, args.momentum, args.device, train_dl_local, test_dl_local,  pre_train_dl, validation_dl))\n",
    "\n",
    "############### Getting gradient similarity matrix\n",
    "clients_backup = [copy.deepcopy(client) for client in clients] # Make a backup of clients' models\n",
    "\n",
    "# --- Added: build mask using dimension of one flattened gradient\n",
    "#    Used to pick random indices for all clients graidients, a small subset for compression\n",
    "#    Assosscation: compress_gradient_with_mask, generate_sparsity_mask, Modified pairwise_angles\n",
    "#    Assosscation: Modified: flatten + sparsify in gradient similarity\n",
    "compression_ratio = 50            # e.g. keep 1/100 of coordinates\n",
    "first_diff = clients[0].get_W()\n",
    "first_flat = flatten(first_diff).detach().cpu().numpy()\n",
    "D = first_flat.shape[0]\n",
    "mask = generate_sparsity_mask(D, compression_ratio)\n",
    "# --- End of Added\n",
    "\n",
    "list_of_dW=[]\n",
    "#w_glob = copy.deepcopy(initial_state_dict)\n",
    "for iteration in range(1): #run the model for 25 iterations before gettin gradients edited\n",
    "    idxs_users = np.arange(args.num_users) #get all users id\n",
    "    print(f'###### Gradient data ROUND {iteration+1} ######')\n",
    "    list_of_dW.clear()\n",
    "    #get global state and train\n",
    "    for idx in idxs_users:\n",
    "        #clients[idx].set_state_dict(copy.deepcopy(w_glob))\n",
    "        oldW = clients[idx].get_W()\n",
    "        loss = clients[idx].train(is_print=False)\n",
    "        newW = clients[idx].get_W()\n",
    "        difference_W = copy.deepcopy(oldW)\n",
    "        subtract_(target=difference_W, minuend=newW, subtrahend=oldW)\n",
    "        \n",
    "        # --- Modified: flatten + sparsify\n",
    "        flat = flatten(difference_W).detach().cpu().numpy()\n",
    "        sparse = compress_gradient_with_mask(flat, mask)\n",
    "        list_of_dW.append(sparse)\n",
    "        # --- End of Modified\n",
    "        \n",
    "        # list_of_dW.append(difference_W) \n",
    "        #-uncomment incase remove the  \"Modified: flatten + sparsify\",\n",
    "        #i.e, remove gradient compression\n",
    "        \n",
    "       \n",
    "    \n",
    "Grad_similarites = pairwise_angles(list_of_dW) \n",
    "clients = [copy.deepcopy(client) for client in clients_backup] # Reset clients using backup\n",
    "\n",
    "######## Getting class-wise weighted data similarity matrix\n",
    "\n",
    "def compute_w(fc1, fc2):\n",
    "    if fc1 == 0 or fc2 == 0:\n",
    "        return None  # Skip positions where fc1 or fc2 is 0\n",
    "    \n",
    "    max_val = max(fc1, fc2)\n",
    "    min_val = min(fc1, fc2)\n",
    "    \n",
    "    w = max_val/min_val\n",
    "    return w\n",
    "\n",
    "def normalize_weights(weights, r):\n",
    "    # Flatten the weights matrix while excluding None values\n",
    "    valid_weights = [w for w in weights.flatten() if w is not None]\n",
    "    \n",
    "    min_weight = min(valid_weights)\n",
    "    max_weight = max(valid_weights)\n",
    "    \n",
    "    # Normalize weights to the range [1, 1 + r]\n",
    "    normalized_weights = np.full_like(weights, None, dtype=np.float64)\n",
    "    for idx in np.ndindex(weights.shape):\n",
    "        if weights[idx] is not None:\n",
    "            normalized_weights[idx] = (weights[idx] - min_weight) / (max_weight - min_weight) * r + 1\n",
    "   \n",
    "    return normalized_weights\n",
    "\n",
    "def calculating_weighted_adjacency(nclients, U_dict, labels, traindata_cls_counts, r):\n",
    "    sim_mat = np.zeros([nclients, nclients])\n",
    "    weights = np.full([nclients, nclients, labels], None)  # 3D array to store weights, initialized with None\n",
    "    \n",
    "    # Compute and store weights\n",
    "    for idx1 in range(nclients):\n",
    "        for idx2 in range(nclients):\n",
    "            if idx1 == idx2:\n",
    "                continue\n",
    "            for l in range(labels):\n",
    "                if l in traindata_cls_counts[idx1] and l in traindata_cls_counts[idx2]:\n",
    "                    f1 = traindata_cls_counts[idx1][l]\n",
    "                    f2 = traindata_cls_counts[idx2][l]\n",
    "                    weights[idx1, idx2, l] = compute_w(f1, f2)  # Parameter b\n",
    "    \n",
    "    # Normalize the weights matrix\n",
    "    normalized_weights = normalize_weights(weights, r)\n",
    "    \n",
    "    # Compute the similarity matrix using normalized weights\n",
    "    for idx1 in range(nclients):\n",
    "        for idx2 in range(nclients):\n",
    "            if idx1 == idx2: \n",
    "                sim_mat[idx1, idx2] = 0\n",
    "                continue\n",
    "            \n",
    "            angles = []\n",
    "            for l in range(labels):\n",
    "                if l in U_dict[idx1] and l in U_dict[idx2]:\n",
    "                    U1 = copy.deepcopy(U_dict[idx1][l])\n",
    "                    U2 = copy.deepcopy(U_dict[idx2][l])\n",
    "                    mul = np.clip(U1.T @ U2, a_min=-1.0, a_max=1.0)\n",
    "                    angle = np.min(np.arccos(mul)) * 180 / np.pi\n",
    "                    weight = normalized_weights[idx1, idx2, l]\n",
    "                    if weight is not None:\n",
    "                        angle = min(angle * weight, 180)\n",
    "                    angles.append(angle)\n",
    "                elif l not in traindata_cls_counts[idx1] and l not in traindata_cls_counts[idx2]:\n",
    "                    angles.append(0)\n",
    "                else:\n",
    "                    angles.append(180)\n",
    "            \n",
    "            sim_mat[idx1, idx2] = sum(angles) / len(angles)\n",
    "    \n",
    "    return sim_mat\n",
    "\n",
    "v=calculating_weighted_adjacency(args.num_users, U_dict, 10,traindata_cls_counts,.5) #label hardcode\n",
    "G= normalize_matrix(Grad_similarites)\n",
    "v= normalize_matrix(v)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Learned row-gates w (first 10 shown): [0.0010 0.0003 0.0006 0.4539 0.0003 0.9995 0.0001 0.0090 0.0001 0.0022\n",
      " 0.0000 0.0000 0.0001 1.0000 1.0000 0.0028 0.0000 0.0017 0.0001 0.0000\n",
      " 0.0005 0.0000 0.0011 0.0013 0.0010 0.0020 0.0101 0.0001 0.0000 0.0004\n",
      " 0.0001 0.0000 0.0001 0.0015 0.0128 0.0003 0.0000 0.0018 0.0000 0.0008\n",
      " 0.0017 0.0001 0.0000 0.0014 0.0061 0.0000 0.0074 0.0006 1.0000 0.0064\n",
      " 0.0014 0.0000 0.0000 0.8918 0.0003 0.0001 0.0000 1.0000 0.0006 0.0033\n",
      " 0.9999 0.9998 0.0008 0.0038 0.0004 1.0000 0.0008 0.0002 0.0109 0.0061\n",
      " 0.0000 0.0170 1.0000 0.0059 0.0003 0.0038 0.0009 0.0000 0.0051 0.0000\n",
      " 0.0010 1.0000 0.0000 0.0008 0.0036 0.0005 0.9997 0.0034 0.8843 0.0002]\n"
     ]
    }
   ],
   "source": [
    "#Fusion-gate: combing data and gradient\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "\n",
    "# ----------------------------------------------------------------------\n",
    "# 1.  MLP that outputs an n-dimensional sigmoid-gated vector w ∈ (0,1)^n\n",
    "# ----------------------------------------------------------------------\n",
    "class RowGateMLP(nn.Module):\n",
    "    def __init__(self, n: int, hidden: int = 128):\n",
    "        \"\"\"\n",
    "        Args:\n",
    "            n (int)     : number of clients  (size of similarity matrices)\n",
    "            hidden (int): hidden layer width\n",
    "        \"\"\"\n",
    "        super().__init__()\n",
    "        d_in = 2 * n * n            # vec(G) ‖ vec(P)\n",
    "        self.net = nn.Sequential(\n",
    "            nn.Linear(d_in, hidden),\n",
    "            nn.ReLU(inplace=True),\n",
    "            nn.Linear(hidden, n),    # logits for each client\n",
    "            nn.Sigmoid()             # element-wise → w ∈ (0,1)^n\n",
    "        )\n",
    "\n",
    "    def forward(self, G: torch.Tensor, P: torch.Tensor) -> torch.Tensor:\n",
    "        flat = torch.cat([G.flatten(), P.flatten()], dim=0)   # (2n²,)\n",
    "        return self.net(flat)                                 # (n,)\n",
    "\n",
    "# ----------------------------------------------------------------------\n",
    "# 2.  Fusion  (upper triangle uses row-gate, mirrored for symmetry)\n",
    "# ----------------------------------------------------------------------\n",
    "def fuse_similarity(G: torch.Tensor,\n",
    "                    P: torch.Tensor,\n",
    "                    gate_net: RowGateMLP):\n",
    "    \"\"\"\n",
    "    Returns\n",
    "        A : symmetric fused similarity  (n×n)\n",
    "        w : vector of row gates         (n,)\n",
    "    Formula (for i<j):\n",
    "        A_ij = w_i G_ij + (1-w_i) P_ij ;  A_ji = A_ij ;  A_ii = 0\n",
    "    \"\"\"\n",
    "    w = gate_net(G, P)                       # (n,)\n",
    "    n = w.size(0)\n",
    "    w_col = w.view(n, 1)                     # (n,1) for broadcasting\n",
    "\n",
    "    mix = w_col * G + (1 - w_col) * P        # apply gate row-wise\n",
    "    upper = torch.triu(mix, diagonal=1)      # keep strict upper triangle\n",
    "    A = upper + upper.T                      # mirror → symmetric, zero diag\n",
    "\n",
    "    return A, w\n",
    "\n",
    "# ----------------------------------------------------------------------\n",
    "# 3.  Row-softmax entropy loss  (no Laplacian term)\n",
    "# ----------------------------------------------------------------------\n",
    "def entropy_loss(A: torch.Tensor) -> torch.Tensor:\n",
    "    n = A.size(0)\n",
    "    soft = F.softmax(A, dim=1)\n",
    "    logt = torch.clamp(soft, 1e-12, 1).log()\n",
    "    return -(soft * logt).sum() / n          # minimise → sharper rows\n",
    "\n",
    "# ----------------------------------------------------------------------\n",
    "# 4.  Training example  (G and v are provided 100×100 NumPy or Torch)\n",
    "# ----------------------------------------------------------------------\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "\n",
    "def to_tensor(mat):\n",
    "    if torch.is_tensor(mat):\n",
    "        return mat.clone().to(device)        # duplicate to avoid in-place edits\n",
    "    return torch.as_tensor(mat, dtype=torch.float32, device=device)\n",
    "\n",
    "G_t = to_tensor(G)          # gradient similarity (100×100)\n",
    "V_t = to_tensor(v)          # data      similarity (100×100)\n",
    "\n",
    "n = G_t.size(0)\n",
    "gate_net = RowGateMLP(n).to(device)\n",
    "optimizer = torch.optim.Adam(gate_net.parameters(), lr=1e-4)\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": 10,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Round 0\n",
      "Breaking HC\n",
      "\n",
      "Adjacency Matrix\n",
      "[[0.0000 0.9999 0.3424 ... 0.3460 0.9998 0.3484]\n",
      " [0.9999 0.0000 1.0000 ... 1.0000 1.0000 1.0000]\n",
      " [0.3424 1.0000 0.0000 ... 0.0094 0.6672 0.0107]\n",
      " ...\n",
      " [0.3460 1.0000 0.0094 ... 0.0000 0.6945 0.0372]\n",
      " [0.9998 1.0000 0.6672 ... 0.6945 0.0000 0.6690]\n",
      " [0.3484 1.0000 0.0107 ... 0.0372 0.6690 0.0000]]\n",
      "\n",
      "Clusters: \n",
      "[[0, 21, 23, 58, 83, 90, 62, 51, 75, 77, 92, 50, 18, 76, 88, 39, 40, 30, 65, 64, 8, 87, 2, 37, 79, 12, 56, 70, 96, 31, 85, 11, 16, 34, 97, 67, 86, 99, 3, 14], [1, 46, 35, 4, 68, 7, 91, 10, 69, 27, 73, 59, 84, 36, 57, 47, 61, 13], [5, 20, 54, 94, 93, 22, 25, 44, 52, 98, 29, 43, 49, 66, 71, 80, 28, 81, 45, 53, 72], [6, 42, 17, 95, 63, 15, 19, 26, 24, 82, 32, 33, 38, 78, 55, 89, 9, 74, 41, 48, 60]]\n",
      "\n",
      "Number of Clusters 4\n",
      "\n",
      "Cluster 0: 40 \n",
      "Cluster 1: 18 \n",
      "Cluster 2: 21 \n",
      "Cluster 3: 21 \n",
      "Clients: Cluster_ID \n",
      "{0: 0, 1: 1, 2: 0, 3: 0, 4: 1, 5: 2, 6: 3, 7: 1, 8: 0, 9: 3, 10: 1, 11: 0, 12: 0, 13: 1, 14: 0, 15: 3, 16: 0, 17: 3, 18: 0, 19: 3, 20: 2, 21: 0, 22: 2, 23: 0, 24: 3, 25: 2, 26: 3, 27: 1, 28: 2, 29: 2, 30: 0, 31: 0, 32: 3, 33: 3, 34: 0, 35: 1, 36: 1, 37: 0, 38: 3, 39: 0, 40: 0, 41: 3, 42: 3, 43: 2, 44: 2, 45: 2, 46: 1, 47: 1, 48: 3, 49: 2, 50: 0, 51: 0, 52: 2, 53: 2, 54: 2, 55: 3, 56: 0, 57: 1, 58: 0, 59: 1, 60: 3, 61: 1, 62: 0, 63: 3, 64: 0, 65: 0, 66: 2, 67: 0, 68: 1, 69: 1, 70: 0, 71: 2, 72: 2, 73: 1, 74: 3, 75: 0, 76: 0, 77: 0, 78: 3, 79: 0, 80: 2, 81: 2, 82: 3, 83: 0, 84: 1, 85: 0, 86: 0, 87: 0, 88: 0, 89: 3, 90: 0, 91: 1, 92: 0, 93: 2, 94: 2, 95: 3, 96: 0, 97: 0, 98: 2, 99: 0}\n",
      "Cluster 0, First Client 0: {2: 16, 6: 102, 7: 75}\n",
      "Cluster 1, First Client 1: {3: 344, 8: 348, 9: 705}\n",
      "Cluster 2, First Client 5: {0: 249, 4: 30, 5: 180}\n",
      "Cluster 3, First Client 6: {0: 184, 1: 712, 5: 75}\n"
     ]
    }
   ],
   "source": [
    "#HR clustering \n",
    "\n",
    "\n",
    "import numpy as np\n",
    "import copy\n",
    "from scipy.sparse.csgraph import connected_components\n",
    "from scipy.sparse import csr_matrix\n",
    "\n",
    "\n",
    "args.cluster_alpha=0.5\n",
    "#1.9\n",
    "args.linkage = 'average' \n",
    "np.set_printoptions(precision=4)\n",
    "\n",
    "cnt = args.num_users\n",
    "for r in range(1):\n",
    "    print(f'Round {r}')\n",
    "    clients_idxs = np.arange(cnt)\n",
    "    \n",
    "    adj_mat=A.detach().cpu().numpy()\n",
    "    #adj_mat=  normalize_matrix(v)*(1-grad_co) + grad_co*normalize_matrix((copy.deepcopy(Grad_similarites)))\n",
    "    #adj_mat=  normalize_matrix((copy.deepcopy(Grad_similarites)))\n",
    "    #adj_mat=  normalize_matrix(v)\n",
    "  \n",
    "    clusters = hierarchical_clustering(copy.deepcopy(adj_mat), thresh=args.cluster_alpha, linkage='average')\n",
    "\n",
    "    cnt+= 10\n",
    "    print('')\n",
    "    print('Adjacency Matrix')\n",
    "    print(adj_mat)\n",
    "    print('')\n",
    "    print('Clusters: ')\n",
    "    print(clusters)\n",
    "    print('')\n",
    "    print(f'Number of Clusters {len(clusters)}')\n",
    "    print('')\n",
    "    for jj in range(len(clusters)):\n",
    "        print(f'Cluster {jj}: {len(clusters[jj])} ')\n",
    "        \n",
    "\n",
    "clients_clust_id = {i:None for i in range(args.num_users)}\n",
    "for i in range(args.num_users):\n",
    "    for j in range(len(clusters)):\n",
    "        if i in clusters[j]:\n",
    "            clients_clust_id[i] = j\n",
    "           # print(i, \" Client in\", \"Cluster \", j, traindata_cls_counts[i] )\n",
    "            break\n",
    "print(f'Clients: Cluster_ID \\n{clients_clust_id}')\n",
    "for k in range(len(clusters)):\n",
    "    print(f\"Cluster {k}, First Client {clusters[k][0]}:\", traindata_cls_counts[clusters[k][0]])\n",
    "\n",
    "\n",
    "\n",
    "import numpy as np\n",
    "from typing import Dict, List\n",
    "\n",
    "def build_similarity_graph_rank_supply_nonzero(\n",
    "        traindata_cls_counts: Dict[int, Dict[int, int]],\n",
    "        clusters: List[List[int]],\n",
    "        num_classes: int = 10,\n",
    "        top_k: int = 2\n",
    ") -> Dict[int, List[int]]:\n",
    "    \"\"\"\n",
    "    Build a directed Cluster-Complementarity Graph.\n",
    "    Demand is high for rare classes; supply is high for abundant classes.\n",
    "\n",
    "    Parameters\n",
    "    ----------\n",
    "    traindata_cls_counts : {client_id: {class: count}}\n",
    "    clusters             : list of clusters (each is a list of client IDs)\n",
    "    num_classes          : total number of classes\n",
    "    top_k                : keep this many outgoing edges per cluster\n",
    "\n",
    "    Returns\n",
    "    -------\n",
    "    similarity_graph : {cluster_id: [target clusters]}\n",
    "    \"\"\"\n",
    "    C, K = len(clusters), num_classes\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 1. Per-client rank over *non-zero* classes\n",
    "    # ----------------------------------------------------------\n",
    "    client_rank = [{} for _ in range(len(traindata_cls_counts))]\n",
    "    for cid, counts in traindata_cls_counts.items():\n",
    "        present = [(cls, cnt) for cls, cnt in counts.items() if cnt > 0]\n",
    "        if not present:\n",
    "            continue                                # client has no data\n",
    "        present.sort(key=lambda x: x[1])            # ascending (rarest first)\n",
    "        for r, (cls, _) in enumerate(present):\n",
    "            client_rank[cid][cls] = r               # rank 0 … m_i-1\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 2. Demand weight  w[p,k]   (large if class k is rare in cluster p)\n",
    "    # ----------------------------------------------------------\n",
    "    w = np.zeros((C, K), dtype=float)\n",
    "    for p, clist in enumerate(clusters):\n",
    "        for cid in clist:\n",
    "            m_i = len(client_rank[cid])             # number of classes present\n",
    "            for cls, r in client_rank[cid].items():\n",
    "                w[p, cls] += (m_i - r)              # larger when rarer\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 3. Supply strength s[q,k] (large if class k is abundant in cluster q)\n",
    "    # ----------------------------------------------------------\n",
    "    s = np.zeros((C, K), dtype=float)\n",
    "    for q, clist in enumerate(clusters):\n",
    "        for cid in clist:\n",
    "            for cls, r in client_rank[cid].items():\n",
    "                s[q, cls] += (r + 1)                # larger when more common\n",
    "        n_q = max(len(clist), 1)\n",
    "        s[q] /= n_q                                 # per-client normalisation\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 4. Complementarity score matrix  A = W · Sᵀ\n",
    "    # ----------------------------------------------------------\n",
    "    score = w @ s.T\n",
    "    np.fill_diagonal(score, -np.inf)                # forbid self-loops\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 5. Keep top-k outgoing edges for each cluster\n",
    "    # ----------------------------------------------------------\n",
    "    graph = {p: np.argsort(-score[p])[:top_k].tolist() for p in range(C)}\n",
    "    return graph\n",
    "similarity_graph = build_similarity_graph_rank_supply_nonzero(\n",
    "    traindata_cls_counts,\n",
    "    clusters,\n",
    "    num_classes=10,\n",
    "    top_k=2\n",
    ")\n",
    "\n",
    "# convert “out-list” format {p:[q1,q2]}  ➜  incoming list  {q:[p1,p2]}\n",
    "H_out = {z: [] for z in range(len(clusters))}\n",
    "for src, tgts in similarity_graph.items():\n",
    "    for t in tgts:\n",
    "        H_out[t].append(src)        # cluster src  ➜  learns from  ➜  t"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'clustering_loss_tiny_penalty' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[6], line 60\u001b[0m\n\u001b[1;32m     58\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m α \u001b[38;5;129;01min\u001b[39;00m alpha_grid:\n\u001b[1;32m     59\u001b[0m     cl \u001b[38;5;241m=\u001b[39m hierarchical_clustering(copy\u001b[38;5;241m.\u001b[39mdeepcopy(adj_mat), thresh\u001b[38;5;241m=\u001b[39mα)\n\u001b[0;32m---> 60\u001b[0m     _, I, P \u001b[38;5;241m=\u001b[39m \u001b[43mclustering_loss_tiny_penalty\u001b[49m(cl, adj_mat, lambda_w, gamma_penalty)\n\u001b[1;32m     61\u001b[0m     records\u001b[38;5;241m.\u001b[39mappend(\u001b[38;5;28mdict\u001b[39m(alpha\u001b[38;5;241m=\u001b[39mα, clusters\u001b[38;5;241m=\u001b[39mcl, intra\u001b[38;5;241m=\u001b[39mI, penalty\u001b[38;5;241m=\u001b[39mP))\n\u001b[1;32m     63\u001b[0m \u001b[38;5;66;03m# Print the raw values collected in records\u001b[39;00m\n",
      "\u001b[0;31mNameError\u001b[0m: name 'clustering_loss_tiny_penalty' is not defined"
     ]
    }
   ],
   "source": [
    "# =============================================================\n",
    "#   Grid–search over hierarchical-clustering threshold α\n",
    "#   with intra-cluster distance and tiny-cluster penalty\n",
    "# =============================================================\n",
    "import copy, numpy as np\n",
    "from scipy.cluster.hierarchy import linkage, fcluster\n",
    "from scipy.spatial.distance  import squareform\n",
    "\n",
    "# 1. Hierarchical clustering\n",
    "def hierarchical_clustering(adj_mat: np.ndarray, thresh: float, linkage_method: str = \"average\"):\n",
    "    condensed = squareform(adj_mat, checks=False)\n",
    "    Z = linkage(condensed, method=linkage_method)\n",
    "    labels = fcluster(Z, t=thresh, criterion=\"distance\")\n",
    "    return [np.where(labels == k)[0].tolist() for k in range(1, labels.max() + 1)]\n",
    "\n",
    "import numpy as np\n",
    "\n",
    "def clustering_loss_with_tiny_penalty(clusters, adjacency_matrix, gamma=1.0, tau=1.0):\n",
    "    \"\"\"\n",
    "    Computes the clustering score using intra-cluster compactness and a tiny-cluster penalty.\n",
    "\n",
    "    Returns:\n",
    "        score:      Intra + TinyPenalty (lower is better)\n",
    "        avg_intra:  Sum of per-cluster compactness\n",
    "        penalty:    Tiny-cluster penalty term\n",
    "    \"\"\"\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 distances ---\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",
    "    # --- Tiny-cluster penalty ---\n",
    "    s_bar = n / K\n",
    "    sigma_s = np.std(sizes, ddof=0)\n",
    "    penalty = np.mean([\n",
    "        np.exp(-max(0, (s_bar - gamma * sigma_s) - s_c) / tau)\n",
    "        for s_c in sizes\n",
    "    ])\n",
    "\n",
    "    score = intra_sum + penalty  # smaller score is better\n",
    "    return score, intra_sum, penalty\n",
    "\n",
    "\n",
    "# 3. Grid search over α\n",
    "alpha_grid = np.linspace(0.05, 1.00, 20)\n",
    "lambda_w = 1.0\n",
    "gamma_penalty = 1.0\n",
    "records = []\n",
    "\n",
    "for α in alpha_grid:\n",
    "    cl = hierarchical_clustering(copy.deepcopy(adj_mat), thresh=α)\n",
    "    _, I, P = clustering_loss_tiny_penalty(cl, adj_mat, lambda_w, gamma_penalty)\n",
    "    records.append(dict(alpha=α, clusters=cl, intra=I, penalty=P))\n",
    "\n",
    "# Print the raw values collected in records\n",
    "header = f\"{'α':>5} | {'#cl':>4} | {'intra':>9} | {'penalty':>9} | {'raw_loss':>9}\"\n",
    "print(header)\n",
    "print(\"-\" * len(header))\n",
    "\n",
    "for r in records:\n",
    "    loss = r['intra'] + r['penalty']\n",
    "    print(f\"{r['alpha']:5.2f} | {len(r['clusters']):4d} | {r['intra']:9.4f} | {r['penalty']:9.4f} | {loss:9.4f}\")\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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tG/fPsXGxmr69On68MMPdeDAAS1cuFCS+VndeeeduvPOOxUfH68VK1Zo1qxZ+uKLL7R161Zt2rQpyxvpxo0bq3Hjxpo3b17KgNd58+apSZMmaQZelyhRQs8884yeeeYZ7dixQzExMZo2bZpee+01nT17Vm+//XamdZUoUUJJSUk6d+6cW7edHTt26Ny5cxmux5CRvHjv5Yfjx4+neyN96b6pU6emlIcNG5aSVPj7+6tVq1ZatWqVli9frquvvjpH8XjzN+L666/X9ddfr5MnT+qnn37SnDlz9P7776tHjx76+++/s9U6EBYWJofDoSNHjngUb0Z/I3Lj/QD4EsZUALngiSeeUEREhKZOneo2DeQ///wjSbrmmmvczk9OTtaqVavSvZaz+8rChQu1atUqtySgc+fOiomJ8bhP97Bhw+Tn56d33nlHhw8fzvTcS7tnXco5u1J6q/Ju2LAh0+cGBASoTZs2mjBhgl5//XVZlqXvvvsu5bjzH3R6n862bt1akvJsRiVvNWnSRJL0008/pTmW3r7c0KxZM509e1Y///xzmmPORDWjT3cjIiJ00003acGCBapbt64WL16c7jS5ZcuW1XXXXafPP/9cXbp00ebNm9PtdpaeW265RWfPntVXX32lr776SmfPnk1ZMDIjNWvW1O23367Y2FhddtllmjdvXpb11K9fX5KZjjW1jz76SJKra1p25cV7z8/PL1utDTlRo0YNWZYly7JS3h/ffvttyr7y5cvrzjvvTNm2LCvNVMrOabGfffZZWZaVaX15+TfC2S3unXfe0bBhw3To0CGtXbs25XixYsUy/Hm2bt1a8fHx2rp1a6Z1eCMn7wfAV5BUALkgJCREjz32mC5evKiJEyem7HeOlVi5cqXb+c8//7z++OOPdK/VsWNH+fn5acqUKTp9+rTbJ6KdO3fW3r179fHHH6tUqVLZ7hZQp04dPfroozpy5Ih69eqlHTt2pDnn3LlzeuWVV9KdQz+1yy+/POWmz9klSTKfkD/99NNpzv/ll19Suo+k5vxEPfXgTOeg4UvXBJCk2267TaGhoRozZoz+/PPPNMfPnDmTMu4iPzlboCZOnKhz586l7D9w4IBee+21PKlz6NChksxUsakHRO/du1evvPKK/P39U+I6f/68li5dmuZG8fTp00pISFBAQEBKMrdw4cI0ny5fvHgx5XVObyBtev7zn/+oWLFi+vjjj/XRRx+ljLVI7fDhw+kmRceOHdP58+ezVZdzSs/o6GhduHBBkukC42zh2Llzp/7+++9sxSzlzXuvTJkyOnLkiNvvRl6KjY1VsWLF1KFDB0lSXFycDh8+nOV4iiFDhqhjx45atmyZbrvttpRWptQOHjyou+66K6X7Z0ZatmwpyUyukJycnLJ/9erVaSYQkMzK9un9fJx/Ny79G5He3wdJKdMo33777eku4nngwIGUcTjZkVvvB8BX0P0JyCX//e9/9fzzz+vDDz/UE088odq1a2v48OGaPn26rr/+eg0ePFhly5bVmjVr9Ouvv6pPnz7pLlQWFhamFi1a6Oeff3YbbCkp5Sbn8OHDuvbaa7OcTz61p59+WufOndPkyZN1+eWXq0uXLmrYsKECAgK0Y8cOLV68WPHx8ekmBqkFBgbq3nvv1XPPPafmzZvr2muvVUJCgr799ltFRkamtM44ffLJJ5o6daqioqJUp04dhYWFKS4uTt9//73KlSuXMsuNZGYP+vLLL3XDDTeod+/eCg4OVqNGjdSnTx+VL19es2bN0g033KAmTZqoZ8+euuKKK3Tu3Dnt2rVLsbGxateuXZY3PLmtW7du+s9//qNPPvlEjRo10rXXXqvz58/riy++UOvWrfXtt9969DpJJkG5dJyA01NPPaUhQ4Zozpw5+uabb9S4cWP17ds3ZZ2K+Ph4vfzyy6pVq5Yks1J6165dVatWLbVu3VrVqlXTqVOn9N133+nAgQN67LHHUgZGDx48WMWLF1eHDh1UvXp1Xbx4UYsWLVJcXJwGDx6c7fEJERER6tKlS8qMSV27dlVERITbOXv37lXr1q115ZVXqnnz5qpcubLi4+P1zTff6OLFi3r00UezrGfQoEGaMmWKFi1apFatWqlLly764YcfdPDgQT377LMaM2aMbrjhBt188816/PHHs7xeXrz3unTponXr1qlfv37q2LGjAgMD1aFDh5Sb/twWGxurRo0apXQXci5u6FyLJCP+/v6aO3eubrjhBs2cOVPz5s1T9+7dVbNmTV24cEFxcXFatmyZLl68mGWrU5s2bdS2bVstXbpUbdu2VadOnbRr1y7NmzdP/fr109dff+12/v/+9z/t3r1bUVFRqlGjhhwOh1auXKmff/5Z7dq1c+se2aVLF33xxRcaOHCgmjVrJj8/P/Xp00eNGjVSz549NXbsWE2cOFF16tRRz549Vb16dcXHx2vbtm1asWKFnn766ZQWrqzk1vsB8Bm2TGQLFEKZrVPh9MYbb1iSrCFDhqTsi4mJsdq3b2+FhoZapUqVsnr37m2tX78+07ncH3vsMUuS1bp16zTHqlevbkmyJk+e7NX38csvv1i33367VadOHSskJMQKCgqyatSoYd10003Wjz/+6HZuRjEmJiZa48aNs6pWrWoFBgZa9erVs1577TVr+/btadZlWLNmjXX33XdbDRs2tEqVKmWFhIRYdevWte6//35r9+7dbte9ePGi9eijj1rVqlWz/P39013j4a+//rLuuOMOq3r16lZgYKBVunRpq1GjRtb9999v/fzzzynnpbdmxqUyW6ciu3PsO+OeOHGiVbNmTSswMNCqVauW9eyzz1pr1661JFkPPPBAhjGk5lynIrPHhg0bUup86aWXrEaNGllBQUFWaGioFRkZaX3zzTdu17xw4YL1/PPPW927d7eqVKliBQYGWhUrVrQiIyPTrI0wdepU65prrrGqV69uBQcHW2XLlrVat25tvf3229bFixez9T04zZw5MyXmmTNnpjl+7NgxKzo62urUqZMVHh5uBQYGWhEREVbPnj2thQsXZrue48ePWw888IBVpUoVKyAgwKpYsWLKmijvvPOOVbFiRatkyZLZvp63772M1qlISEiw7rrrLis8PNwqVqyY2zmervmSlcTERCssLMy67777UvbdfPPNVo0aNbJ9jeTkZOvLL7+0rrvuOisiIsIKDAy0ihcvbjVs2NC6//77rbi4OLfzM1pb5vDhw9aQIUOsMmXKWCEhIVabNm2shQsXpvse+uyzz6xBgwZZtWvXtooXL26VLFnSatq0qfXCCy9Yp06dcrvu/v37rUGDBlnlypVL+Xle+n5ctGiR1a9fP6t8+fJWQECAValSJatt27bWxIkT3f7mZPR+dsrN9wPgCxyWlUXnSQCAV9577z3dddddmjp1qu655x67wwEAIM+QVABADh04cEAVK1Z0m0Vm7969at++vfbs2aMdO3ZkOAUsAABFAWMqACCHnnvuOc2fPz9lwa7du3fru+++U0JCgqKjo0koAABFHkkFAORQz549FRcXp/nz5+vYsWMKDg5W48aNNWLEiDSzHgEAUBTR/QkAAABAjrBOBQAAAIAcIakA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SZUrm1aJtm1d+595RvroI+mvv/IvRk/4ZEtFs2bN9PPPP6tixYoqViz9HmAJCQlq0KCB4uLiFBoams8RIr/xevseXnPfwuvtW3i9fUtBf72Tk5O1e/duNWjQQP7+rlvv9FopJKlcOcnPL22rxKFDaVsvChKfbKnIjpMnT6pkyZJpmqpQNPF6+x5ec9/C6+1beL19S1F8vVu3NtPITp3q2teggXTttdKkSfbFlRmfbKkAAAAACqpRo6QhQ6SWLU0XqHfeMdPJDh9ud2QZI6kAAAAACpDBg6X4eOmpp6T9+6WGDaXvv5eqV7c7soyRVGQgKChI48ePz7C/G4oWXm/fw2vuW3i9fQuvt28pqq/3iBHmUVgwpgIAAABAjrD4HQAAAIAcIakAAAAAkCMkFQAAAAByxKeTiqlTp6pmzZoKDg5WixYttGLFikzPj42NVYsWLRQcHKxatWpp2rRp+RQpcoMnr/ecOXN09dVXq3z58goLC1Pbtm21cOHCfIwWOeXp+9tp1apV8vf3V9OmTfM2QOQ6T1/z8+fPa8yYMapevbqCgoJUu3ZtffDBB/kULXLK09f7k08+UZMmTVS8eHGFh4frtttuU3x8fD5Fi5xYvny5+vXrp4iICDkcDs2dOzfL53DPZgPLR3322WdWQECA9e6771pxcXHWAw88YJUoUcLatWtXuudv377dKl68uPXAAw9YcXFx1rvvvmsFBARYX375ZT5HDm94+no/8MAD1vPPP2/9/PPP1pYtW6zRo0dbAQEB1q+//prPkcMbnr7eTsePH7dq1aplde/e3WrSpEn+BItc4c1rfs0111itW7e2Fi1aZO3YscNau3attWrVqnyMGt7y9PVesWKFVaxYMeu1116ztm/fbq1YscK68sorreuuuy6fI4c3vv/+e2vMmDHWV199ZUmyvv7660zP557NHj6bVLRq1coaPny4274rrrjCevzxx9M9/9FHH7WuuOIKt31333231aZNmzyLEbnH09c7PQ0aNLAmTJiQ26EhD3j7eg8ePNh68sknrfHjx5NUFDKevuY//PCDVbJkSSs+Pj4/wkMu8/T1fvHFF61atWq57Xv99detKlWq5FmMyBvZSSq4Z7OHT3Z/unDhgtavX6/u3bu77e/evbt++umndJ+zevXqNOf36NFD69at08WLF/MsVuScN6/3pZKTk5WQkKAy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xzP9mUjSvn37NGjQIIWGhqpkyZIaPHiwDhw4kOXznN566y3169dPpUqVctt/4sQJDR8+XOXLl1doaKi6deumzZs365dffpHD4dCOHTtSzq1YsaKuvvpqTZs2Ldv1AkBB4G93AACQl3r37i0/Pz8tX748w3N27typPn36qGPHjvrggw9UqlQp7d27VwsWLNCFCxcUHh6uBQsWqGfPnrrjjjtSPol23pjv27dPZcuW1XPPPafy5cvr6NGjmjlzplq3bq0NGzbo8ssvd6vviSeeUPv27fXee+/p5MmTeuyxx9SvXz9t3rxZfn5+kqT3339fd911lyIjIzVt2jRVqFBBW7Zs0R9//JFynZiYGPXs2VOtW7fWtGnTVLJkSX322WcaPHiwzpw5kyZJcMrq+3EaMGCABg8erDvuuEObNm3S6NGjJUkffPCB23lz587VihUrNG7cOFWqVEkVKlRQXFyc2rVrl5LYVapUSQsXLtT999+vI0eOaPz48ZKkF154QdHR0XryySfVqVMnXbx4UX/99Ve6SUBW8ViWpeuuu05LlizR6NGj1bFjR23cuFHjx4/X6tWrtXr1agUFBaX7Mzl79qy6deumffv2adKkSapXr57mz5+vwYMHp3v+pfbs2aNNmzbpnnvucdt/8eJFdevWTVu2bNHLL7+siIgIPf7447ruuuvUp08ftWzZUjVr1nR7TlRUlEaPHq3jx4+nSVAAoMCyAKAQmz59uiXJ+uWXXzI8p2LFilb9+vVTtsePH2+l/vP35ZdfWpKs3377LcNrHD582JJkjR8/PsuYEhMTrQsXLlh169a1HnrooZT9MTExliSrd+/ebud/8cUXliRr9erVlmVZVkJCghUWFmZ16NDBSk5OzrCeK664wmrWrJl18eJFt/19+/a1wsPDraSkJK++H+fP54UXXnDbP2LECCs4ONgtJklWyZIlraNHj7qd26NHD6tKlSrWiRMn3Pbfe++9VnBwcMr5ffv2tZo2bZphnJ7Es2DBgnTP+/zzzy1J1jvvvJOyLzIy0oqMjEzZfuuttyxJ1jfffOP23LvuusuSZE2fPj3TGJ11rFmzxm2/83frrbfeStm3du1aS5IVHBxsPf/882mutWjRIkuS9cMPP2RaJwAUJHR/AlDkWZaV6fGmTZsqMDBQ//3vfzVz5kxt377do+snJibq2WefVYMGDRQYGCh/f38FBgZq69at2rx5c5rzr7nmGrftxo0bS5J27dolSfrpp5908uRJjRgxIsNZqrZt26a//vpL//nPf1JicD569+6t/fv3p9ulyhPpxXnu3Lk0M2p16dJFpUuXTtk+d+6clixZov79+6t48eJpYjt37pzWrFkjSWrVqpV+//13jRgxQgsXLtTJkye9jmfp0qWSlKaF5oYbblCJEiW0ZMmSDK8dExOj0NDQNHXcfPPNGT4ntX379kmSKlSo4Lb/p59+kiRdf/31KftatWqlUqVK6dy5c7rhhhvSXMt5jb1792arbgAoCEgqABRpp0+fVnx8vCIiIjI8p3bt2lq8eLEqVKigkSNHqnbt2qpdu3a6/frTM2rUKI0dO1bXXXedvv32W61du1a//PKLmjRpku6g5bJly7ptO7vkOM89fPiwJDOgPCPOgecPP/ywAgIC3B4jRoyQpBxPwZpVnE7h4eFu2/Hx8UpMTNQbb7yRJrbevXu7xTZ69Gi99NJLWrNmjXr16qWyZcuqa9euWrduncfxxMfHy9/fP003LofDoUqVKik+Pj7D7zU+Pl4VK1ZMs79SpUoZPic1ZwzBwcFu+xMSEhQQEJAm2WjevHm6XZ9SX8NXB7wDKJwYUwGgSJs/f76SkpKyXJOgY8eO6tixo5KSkrRu3Tq98cYbevDBB1WxYkXdeOONmT73448/1q233qpnn33Wbf+RI0e86hPvvCnes2dPhueUK1dOkrkpT/0peGqXjuXIK5e2ppQuXVp+fn4aMmSIRo4cme5znDfT/v7+GjVqlEaNGqXjx49r8eLFeuKJJ9SjRw/9+++/Kl68eLbjKFu2rBITE3X48GG3xMKyLB04cEBXXXVVps/9+eef0+zP7kBt5+tx9OhRtyQrIiJCFy9e1OnTp1WiRImU/bt27Uo3oXBeI/U1AaAwoKUCQJG1e/duPfzwwypZsqTuvvvubD3Hz89PrVu3Tpml6Ndff5WU8af0krmpvnQA8Pz5873uvtKuXTuVLFlS06ZNy7Dr1uWXX666devq999/V8uWLdN9ZLZWQmbfT04VL15cnTt31oYNG9S4ceN0Y7u01UGSSpUqpYEDB2rkyJE6evSodu7c6VG9Xbt2lWSSvNS++uornT59OuV4ejp37qyEhATNmzfPbf+nn36arbqvuOIKSdI///zjtr9FixaSpFWrVqXsW7lypf755x9t3Lgx3Ws5u981aNAgW3UDQEFASwWAIuGPP/5I6bd/6NAhrVixQtOnT5efn5++/vrrNF1iUps2bZqWLl2qPn36qFq1ajp37lzKjELdunWTJIWGhqp69er65ptv1LVrV5UpU0blypVTjRo11LdvX82YMUNXXHGFGjdurPXr1+vFF1/MtPtSZi677DK9/PLLuvPOO9WtWzfdddddqlixorZt26bff/9dU6ZMkSS9/fbb6tWrl3r06KFhw4apcuXKOnr0qDZv3qxff/1Vs2fPzrCOzL6f3PDaa6+pQ4cO6tixo+655x7VqFFDCQkJ2rZtm7799tuU8Q/9+vVTw4YN1bJlS5UvX167du3Sq6++qurVq6tu3boe1Xn11VerR48eeuyxx3Ty5Em1b98+ZfanZs2aaciQIRk+99Zbb9XkyZN166236plnnlHdunX1/fffa+HChdmqu3Xr1goJCdGaNWvcxmX07dtXjRo10kMPPaTPPvtM5cuX16hRo9SmTRutWbNGX375pQYOHOh2rTVr1qhs2bJq1KiRR98/ANjK5oHiAJAjztmfnI/AwECrQoUKVmRkpPXss89ahw4dSvOcS2d/Wr16tdW/f3+revXqVlBQkFW2bFkrMjLSmjdvntvzFi9ebDVr1swKCgqyJFlDhw61LMuyjh07Zt1xxx1WhQoVrOLFi1sdOnSwVqxYkWaGIefsT7Nnz3a77o4dO9KdYej777+3IiMjrRIlSljFixe3GjRokGa2oN9//90aNGiQVaFCBSsgIMCqVKmS1aVLF2vatGlZ/uwy+n6cP5/Dhw+n+7PesWNHyj5J1siRI9O9/o4dO6zbb7/dqly5shUQEGCVL1/eateunfX000+nnPPyyy9b7dq1s8qVK2cFBgZa1apVs+644w5r586dKed4Es/Zs2etxx57zKpevboVEBBghYeHW/fcc4917Ngxt+de+tpYlmXt2bPHGjBggHXZZZdZoaGh1oABA6yffvopW7M/WZZlDRkyxGrQoEGa/bt27bIGDBhghYWFWYGBgVZUVJR15MgR67nnnrNKliyZ8nO3LMtKTk62qlevbt13331Z1gcABYnDsrKYFgUAAGRp3bp1uuqqq7RmzRq1bt3aq2ssWbJE3bt3159//pnSpQoACgOSCgAAcsngwYN1+vRpfffdd149v3PnzqpTp47efffdXI4MAPIWA7UBAMglL7/8sq666iolJCR4/Nxjx44pMjJSzzzzTB5EBgB5i5YKAAAAADlCSwUAAACAHCGpAAAAAJAjJBUAAAAAcoSkAgAAAECOkFQAAAAAyBGSCgAAAAA5QlIBAAAAIEdIKgAAAADkyP8BsFgL3vHx4t0AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 800x600 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "--- Plotted values (α, raw_loss, #clusters) ---\n",
      "α = 0.050 | Loss = 1.0000 | #Clusters = 20\n",
      "α = 0.100 | Loss = 1.1204 | #Clusters = 17\n",
      "α = 0.150 | Loss = 1.3673 | #Clusters = 13\n",
      "α = 0.200 | Loss = 1.4248 | #Clusters = 12\n",
      "α = 0.250 | Loss = 1.4248 | #Clusters = 12\n",
      "α = 0.300 | Loss = 1.6478 | #Clusters = 9\n",
      "α = 0.350 | Loss = 1.8995 | #Clusters = 7\n",
      "α = 0.400 | Loss = 1.9634 | #Clusters = 5\n",
      "α = 0.450 | Loss = 1.9120 | #Clusters = 4\n",
      "α = 0.500 | Loss = 1.7351 | #Clusters = 3\n",
      "α = 0.550 | Loss = 1.4576 | #Clusters = 1\n",
      "α = 0.600 | Loss = 1.4576 | #Clusters = 1\n",
      "α = 0.650 | Loss = 1.4576 | #Clusters = 1\n",
      "α = 0.700 | Loss = 1.4576 | #Clusters = 1\n",
      "α = 0.750 | Loss = 1.4576 | #Clusters = 1\n",
      "α = 0.800 | Loss = 1.4576 | #Clusters = 1\n",
      "α = 0.850 | Loss = 1.4576 | #Clusters = 1\n",
      "α = 0.900 | Loss = 1.4576 | #Clusters = 1\n",
      "α = 0.950 | Loss = 1.4576 | #Clusters = 1\n",
      "α = 1.000 | Loss = 1.4576 | #Clusters = 1\n",
      "\n",
      "✅ Plot saved at: plots/alpha_vs_raw_loss_and_clusters.png\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import copy\n",
    "from scipy.cluster.hierarchy import linkage, fcluster\n",
    "from scipy.spatial.distance import squareform\n",
    "\n",
    "# --- Make sure your adj_mat is already defined ---\n",
    "\n",
    "# 1. Hierarchical clustering\n",
    "def hierarchical_clustering(adj_mat: np.ndarray, thresh: float, linkage_method: str = \"average\"):\n",
    "    condensed = squareform(adj_mat, checks=False)\n",
    "    Z = linkage(condensed, method=linkage_method)\n",
    "    labels = fcluster(Z, t=thresh, criterion=\"distance\")\n",
    "    return [np.where(labels == k)[0].tolist() for k in range(1, labels.max() + 1)]\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",
    "    for cluster in clusters:\n",
    "        size = len(cluster)\n",
    "        if size == 0: continue\n",
    "        dist = sum(adjacency_matrix[i, j] for i in cluster for j in cluster)\n",
    "        intra_sum += dist / (size * size)\n",
    "\n",
    "    s_bar = n / K\n",
    "    sigma_s = np.std(sizes, ddof=0)\n",
    "    penalty = np.mean([\n",
    "        np.exp(-max(0, (s_bar - gamma * sigma_s) - s_c) / tau)\n",
    "        for s_c in sizes\n",
    "    ])\n",
    "    return intra_sum + penalty, intra_sum, penalty\n",
    "\n",
    "# 3. Grid search over α\n",
    "alpha_grid = np.linspace(0.05, 1.00, 20)\n",
    "gamma_penalty = 1.0\n",
    "tau_penalty = 1.0\n",
    "records = []\n",
    "\n",
    "for α in alpha_grid:\n",
    "    clusters = hierarchical_clustering(copy.deepcopy(adj_mat), thresh=α)\n",
    "    score, intra, penalty = clustering_loss_with_tiny_penalty(clusters, adj_mat, gamma_penalty, tau_penalty)\n",
    "    records.append(dict(alpha=α, clusters=clusters, score=score))\n",
    "\n",
    "# 4. Prepare for plot\n",
    "alphas = [r[\"alpha\"] for r in records]\n",
    "losses = [r[\"score\"] for r in records]\n",
    "n_clusters = [len(r[\"clusters\"]) for r in records]\n",
    "\n",
    "# 5. Plot\n",
    "os.makedirs(\"plots\", exist_ok=True)\n",
    "fig, ax1 = plt.subplots(figsize=(8, 6))\n",
    "\n",
    "# Bar: number of clusters (right Y-axis)\n",
    "ax2 = ax1.twinx()\n",
    "ax2.bar(alphas, n_clusters, width=0.03, alpha=0.5, color='blue')\n",
    "ax2.set_ylabel(\"Number of Clusters\", color='blue', fontsize=12)\n",
    "ax2.tick_params(axis='y', labelcolor='blue')\n",
    "ax2.set_ylim(0, max(n_clusters) + 5)\n",
    "ax2.set_yticks(range(0, max(n_clusters) + 1, 5))\n",
    "\n",
    "# Line: raw loss (left Y-axis)\n",
    "ax1.plot(alphas, losses, 'x--', color='red', linewidth=2, markersize=6)\n",
    "ax1.set_xlabel(\"Distance threshold (α)\", fontsize=12)\n",
    "ax1.set_ylabel(\"Clustering Loss\", color='red', fontsize=12)\n",
    "ax1.tick_params(axis='y', labelcolor='red')\n",
    "ax1.set_title(\"Raw Clustering Loss vs α with #Clusters\", fontsize=14)\n",
    "\n",
    "# Save\n",
    "plt.tight_layout()\n",
    "plot_path = \"plots/alpha_vs_raw_loss_and_clusters.png\"\n",
    "plt.savefig(plot_path, dpi=300)\n",
    "plt.show()\n",
    "\n",
    "# 6. Print raw values used in the plot\n",
    "print(\"\\n--- Plotted values (α, raw_loss, #clusters) ---\")\n",
    "for a, l, k in zip(alphas, losses, n_clusters):\n",
    "    print(f\"α = {a:.3f} | Loss = {l:.4f} | #Clusters = {k}\")\n",
    "\n",
    "print(f\"\\n✅ Plot saved at: {plot_path}\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Cluster 0, First Client 0: {4: 21, 7: 1, 8: 214}\n",
      "Cluster 1, First Client 6: {2: 153, 3: 79, 9: 123}\n",
      "Cluster 2, First Client 7: {5: 368, 6: 118, 9: 48}\n",
      "Cluster 3, First Client 19: {1: 475, 2: 72, 8: 28}\n"
     ]
    }
   ],
   "source": [
    "for k in range(len(clusters)):\n",
    "    print(f\"Cluster {k}, First Client {clusters[k][0]}:\", traindata_cls_counts[clusters[k][0]])\n",
    "     "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Goes to utility\n",
    "def clone_encoder(encoder_template: nn.Module) -> nn.Module:\n",
    "    \"\"\"\n",
    "    Return a fresh deep copy of the encoder (with the same structure + random weights).\n",
    "    \"\"\"\n",
    "    return copy.deepcopy(encoder_template).to(next(encoder_template.parameters()).device)\n",
    "\n",
    "\n",
    "# ------------------------------------------------------------\n",
    "# Robust FedAvg over a list of encoders (any architecture)\n",
    "# ------------------------------------------------------------\n",
    "def fedavg_encoder(encoder_list: list[nn.Module]) -> nn.Module:\n",
    "    \"\"\"\n",
    "    Return a copy of encoder_list[0] whose parameters are the\n",
    "    element-wise average of all encoders in the list.\n",
    "    \"\"\"\n",
    "    n = len(encoder_list)\n",
    "    assert n > 0, \"encoder_list must be non-empty\"\n",
    "\n",
    "    # clone first encoder as template\n",
    "    avg_enc = copy.deepcopy(encoder_list[0])\n",
    "    avg_sd  = avg_enc.state_dict()           # OrderedDict of tensors\n",
    "\n",
    "    with torch.no_grad():\n",
    "        for k in avg_sd.keys():\n",
    "            stacked = torch.stack([enc.state_dict()[k] for enc in encoder_list], dim=0)\n",
    "            avg_sd[k].copy_(stacked.mean(dim=0))\n",
    "\n",
    "    avg_enc.load_state_dict(avg_sd)\n",
    "    return avg_enc\n",
    "\n",
    "\n",
    "def make_combined_model_from_single(single_model: SimpleCNNMNIST2,\n",
    "                                    cluster_enc: nn.Module,\n",
    "                                    num_classes: int = 10) -> CombinedModelMNIST:\n",
    "    \"\"\"\n",
    "    single_model : an *instance* of SimpleCNNMNIST2 (for architecture reference)\n",
    "    cluster_enc  : averaged encoder to use as PRIMARY encoder\n",
    "    \"\"\"\n",
    "    # fresh secondary encoder (same dim 84-D) – random init\n",
    "    sec_enc = clone_encoder(single_model.encoder)\n",
    "    # empty classifier state-dict with correct input dim 168\n",
    "    new_clf = nn.Linear(168, num_classes)\n",
    "    clf_sd  = new_clf.state_dict()          # random weights\n",
    "    return CombinedModelMNIST(cluster_enc, sec_enc, clf_sd, num_classes)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "def switch_cluster_to_combined(\n",
    "    clients, \n",
    "    cluster_ids, \n",
    "    w_glob_per_cluster, \n",
    "    cluster_id, \n",
    "    num_classes=10\n",
    "):\n",
    "    \"\"\"\n",
    "    Initialize combined model for a cluster using FedAvg on primary encoders.\n",
    "    \n",
    "    clients             : list[Client_ClusterFL]\n",
    "    cluster_ids         : list[int] – client indices in the SAME cluster\n",
    "    w_glob_per_cluster  : list of cluster-level global state_dicts\n",
    "    cluster_id          : int – ID of the current cluster\n",
    "    num_classes         : int – number of output classes\n",
    "    \"\"\"\n",
    "    # 1. collect trained encoders from clients\n",
    "    encoders = [copy.deepcopy(clients[i].net.encoder).to(clients[i].device)\n",
    "                for i in cluster_ids]\n",
    "\n",
    "    # 2. FedAvg on primary encoder\n",
    "    cluster_enc = fedavg_encoder(encoders)\n",
    "\n",
    "    # 3. use the first client’s model as a template\n",
    "    single_template = clients[cluster_ids[0]].net\n",
    "\n",
    "    # 4. build the combined model (primary + frozen secondary + classifier)\n",
    "    combined = make_combined_model_from_single(\n",
    "        single_model=single_template,\n",
    "        cluster_enc=cluster_enc,\n",
    "        num_classes=num_classes\n",
    "    )\n",
    "\n",
    "    # 5. get state_dict of combined model\n",
    "    combined_sd = combined.state_dict()\n",
    "\n",
    "    # 6. assign combined model to all clients in the cluster\n",
    "    for i in cluster_ids:\n",
    "        clients[i].net = copy.deepcopy(combined)\n",
    "        clients[i].net.load_state_dict(combined_sd)\n",
    "\n",
    "    # 7. store cluster-level global model\n",
    "    w_glob_per_cluster[cluster_id] = copy.deepcopy(combined_sd)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      ">>> All clusters switched to CombinedModelMNIST\n"
     ]
    }
   ],
   "source": [
    "w_glob_per_cluster = [None] * len(clusters)\n",
    "\n",
    "for z, clist in enumerate(clusters):\n",
    "    switch_cluster_to_combined(\n",
    "        clients=clients,\n",
    "        cluster_ids=clist,\n",
    "        w_glob_per_cluster=w_glob_per_cluster,\n",
    "        cluster_id=z,\n",
    "        num_classes=10\n",
    "    )\n",
    "print(\">>> All clusters switched to CombinedModelMNIST\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "##### ROUND 1 #####\n",
      "---- ROUND STATS ----\n",
      "avg train loss : 0.4236\n",
      "avg best PRE   : 0.00%\n",
      "avg best POST  : 15.89%\n",
      "avg best ANY   : 15.89%\n",
      "\n",
      "Client   0 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   1 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   2 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   3 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   4 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   5 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   6 | best_pre=  0.00 | best_post= 98.60 | best_any= 98.60\n",
      "Client   7 | best_pre=  0.00 | best_post= 98.53 | best_any= 98.53\n",
      "Client   8 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client   9 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  10 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  11 | best_pre=  0.00 | best_post= 41.03 | best_any= 41.03\n",
      "Client  12 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  13 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  14 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  15 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  16 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  17 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  18 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  19 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  20 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  21 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  22 | best_pre=  0.00 | best_post= 93.03 | best_any= 93.03\n",
      "Client  23 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  24 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  25 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  26 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  27 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  28 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  29 | best_pre=  0.00 | best_post= 96.17 | best_any= 96.17\n",
      "Client  30 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  31 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  32 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  33 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  34 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  35 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  36 | best_pre=  0.00 | best_post= 98.77 | best_any= 98.77\n",
      "Client  37 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  38 | best_pre=  0.00 | best_post= 96.20 | best_any= 96.20\n",
      "Client  39 | best_pre=  0.00 | best_post= 66.67 | best_any= 66.67\n",
      "Client  40 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  41 | best_pre=  0.00 | best_post= 71.03 | best_any= 71.03\n",
      "Client  42 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  43 | best_pre=  0.00 | best_post= 94.77 | best_any= 94.77\n",
      "Client  44 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  45 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  46 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  47 | best_pre=  0.00 | best_post= 69.27 | best_any= 69.27\n",
      "Client  48 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  49 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  50 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  51 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  52 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  53 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  54 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  55 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  56 | best_pre=  0.00 | best_post= 33.33 | best_any= 33.33\n",
      "Client  57 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  58 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  59 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  60 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  61 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  62 | best_pre=  0.00 | best_post= 66.63 | best_any= 66.63\n",
      "Client  63 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  64 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  65 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  66 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  67 | best_pre=  0.00 | best_post= 34.20 | best_any= 34.20\n",
      "Client  68 | best_pre=  0.00 | best_post= 96.47 | best_any= 96.47\n",
      "Client  69 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  70 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  71 | best_pre=  0.00 | best_post= 95.53 | best_any= 95.53\n",
      "Client  72 | best_pre=  0.03 | best_post= 65.10 | best_any= 65.10\n",
      "Client  73 | best_pre=  0.00 | best_post= 88.97 | best_any= 88.97\n",
      "Client  74 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  75 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  76 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  77 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  78 | best_pre=  0.00 | best_post= 90.67 | best_any= 90.67\n",
      "Client  79 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  80 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  81 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  82 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  83 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  84 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  85 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  86 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  87 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  88 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  89 | best_pre=  0.00 | best_post= 94.03 | best_any= 94.03\n",
      "Client  90 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  91 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  92 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  93 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  94 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  95 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  96 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  97 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  98 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "Client  99 | best_pre=  0.00 | best_post=  0.00 | best_any=  0.00\n",
      "\n",
      "##### ROUND 2 #####\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[9], line 91\u001b[0m\n\u001b[1;32m     88\u001b[0m loss_buf\u001b[38;5;241m.\u001b[39mappend(loss)\n\u001b[1;32m     90\u001b[0m \u001b[38;5;66;03m# ---- metrics after train\u001b[39;00m\n\u001b[0;32m---> 91\u001b[0m post_loss, post_acc \u001b[38;5;241m=\u001b[39m \u001b[43mclients\u001b[49m\u001b[43m[\u001b[49m\u001b[43muid\u001b[49m\u001b[43m]\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43meval_test\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     92\u001b[0m client_metric[uid][\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mbest_post\u001b[39m\u001b[38;5;124m\"\u001b[39m] \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mmax\u001b[39m(client_metric[uid][\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mbest_post\u001b[39m\u001b[38;5;124m\"\u001b[39m], post_acc)\n\u001b[1;32m     93\u001b[0m client_metric[uid][\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mbest_any\u001b[39m\u001b[38;5;124m\"\u001b[39m]  \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mmax\u001b[39m(client_metric[uid][\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mbest_any\u001b[39m\u001b[38;5;124m\"\u001b[39m],\n\u001b[1;32m     94\u001b[0m                                       pre_acc, post_acc)\n",
      "File \u001b[0;32m~/Misc/PACFLComboNB/Flag/src/client/client_cluster_fl.py:200\u001b[0m, in \u001b[0;36mClient_ClusterFL.eval_test\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    198\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m data, target \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mldr_test:\n\u001b[1;32m    199\u001b[0m     data, target \u001b[38;5;241m=\u001b[39m data\u001b[38;5;241m.\u001b[39mto(\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mdevice), target\u001b[38;5;241m.\u001b[39mto(\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mdevice)\n\u001b[0;32m--> 200\u001b[0m     output \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mnet\u001b[49m\u001b[43m(\u001b[49m\u001b[43mdata\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    201\u001b[0m     test_loss \u001b[38;5;241m+\u001b[39m\u001b[38;5;241m=\u001b[39m F\u001b[38;5;241m.\u001b[39mcross_entropy(output, target, reduction\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124msum\u001b[39m\u001b[38;5;124m'\u001b[39m)\u001b[38;5;241m.\u001b[39mitem()  \u001b[38;5;66;03m# sum up batch loss\u001b[39;00m\n\u001b[1;32m    202\u001b[0m     pred \u001b[38;5;241m=\u001b[39m output\u001b[38;5;241m.\u001b[39mdata\u001b[38;5;241m.\u001b[39mmax(\u001b[38;5;241m1\u001b[39m, keepdim\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mTrue\u001b[39;00m)[\u001b[38;5;241m1\u001b[39m]  \u001b[38;5;66;03m# get the index of the max log-probability\u001b[39;00m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/module.py:1532\u001b[0m, in \u001b[0;36mModule._wrapped_call_impl\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1530\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_compiled_call_impl(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)  \u001b[38;5;66;03m# type: ignore[misc]\u001b[39;00m\n\u001b[1;32m   1531\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[0;32m-> 1532\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_call_impl\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/module.py:1541\u001b[0m, in \u001b[0;36mModule._call_impl\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1536\u001b[0m \u001b[38;5;66;03m# If we don't have any hooks, we want to skip the rest of the logic in\u001b[39;00m\n\u001b[1;32m   1537\u001b[0m \u001b[38;5;66;03m# this function, and just call forward.\u001b[39;00m\n\u001b[1;32m   1538\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m (\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_pre_hooks\n\u001b[1;32m   1539\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_backward_hooks\n\u001b[1;32m   1540\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_forward_pre_hooks):\n\u001b[0;32m-> 1541\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mforward_call\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1543\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m   1544\u001b[0m     result \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m\n",
      "File \u001b[0;32m~/Misc/PACFLComboNB/Flag/src/models/model.py:327\u001b[0m, in \u001b[0;36mCombinedModelMNIST.forward\u001b[0;34m(self, x)\u001b[0m\n\u001b[1;32m    325\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mforward\u001b[39m(\u001b[38;5;28mself\u001b[39m, x):\n\u001b[1;32m    326\u001b[0m     f1 \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mown_encoder(x)           \u001b[38;5;66;03m# Could be shape [B, 84] or [B, 84, 1, 1]\u001b[39;00m\n\u001b[0;32m--> 327\u001b[0m     f2 \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43msecondary_encoder\u001b[49m\u001b[43m(\u001b[49m\u001b[43mx\u001b[49m\u001b[43m)\u001b[49m     \u001b[38;5;66;03m# Could also be 4D if not flattened internally\u001b[39;00m\n\u001b[1;32m    329\u001b[0m     \u001b[38;5;66;03m# --- Ensure both are 2D: [batch_size, feature_dim]\u001b[39;00m\n\u001b[1;32m    330\u001b[0m     f1_flat \u001b[38;5;241m=\u001b[39m f1\u001b[38;5;241m.\u001b[39mview(f1\u001b[38;5;241m.\u001b[39msize(\u001b[38;5;241m0\u001b[39m), \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m) \u001b[38;5;28;01mif\u001b[39;00m f1\u001b[38;5;241m.\u001b[39mdim() \u001b[38;5;241m>\u001b[39m \u001b[38;5;241m2\u001b[39m \u001b[38;5;28;01melse\u001b[39;00m f1\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/module.py:1532\u001b[0m, in \u001b[0;36mModule._wrapped_call_impl\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1530\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_compiled_call_impl(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)  \u001b[38;5;66;03m# type: ignore[misc]\u001b[39;00m\n\u001b[1;32m   1531\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[0;32m-> 1532\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_call_impl\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/module.py:1541\u001b[0m, in \u001b[0;36mModule._call_impl\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1536\u001b[0m \u001b[38;5;66;03m# If we don't have any hooks, we want to skip the rest of the logic in\u001b[39;00m\n\u001b[1;32m   1537\u001b[0m \u001b[38;5;66;03m# this function, and just call forward.\u001b[39;00m\n\u001b[1;32m   1538\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m (\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_pre_hooks\n\u001b[1;32m   1539\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_backward_hooks\n\u001b[1;32m   1540\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_forward_pre_hooks):\n\u001b[0;32m-> 1541\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mforward_call\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1543\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m   1544\u001b[0m     result \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/container.py:217\u001b[0m, in \u001b[0;36mSequential.forward\u001b[0;34m(self, input)\u001b[0m\n\u001b[1;32m    215\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mforward\u001b[39m(\u001b[38;5;28mself\u001b[39m, \u001b[38;5;28minput\u001b[39m):\n\u001b[1;32m    216\u001b[0m     \u001b[38;5;28;01mfor\u001b[39;00m module \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mself\u001b[39m:\n\u001b[0;32m--> 217\u001b[0m         \u001b[38;5;28minput\u001b[39m \u001b[38;5;241m=\u001b[39m \u001b[43mmodule\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43minput\u001b[39;49m\u001b[43m)\u001b[49m\n\u001b[1;32m    218\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28minput\u001b[39m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/module.py:1532\u001b[0m, in \u001b[0;36mModule._wrapped_call_impl\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1530\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_compiled_call_impl(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)  \u001b[38;5;66;03m# type: ignore[misc]\u001b[39;00m\n\u001b[1;32m   1531\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[0;32m-> 1532\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_call_impl\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/module.py:1541\u001b[0m, in \u001b[0;36mModule._call_impl\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1536\u001b[0m \u001b[38;5;66;03m# If we don't have any hooks, we want to skip the rest of the logic in\u001b[39;00m\n\u001b[1;32m   1537\u001b[0m \u001b[38;5;66;03m# this function, and just call forward.\u001b[39;00m\n\u001b[1;32m   1538\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m (\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_pre_hooks\n\u001b[1;32m   1539\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_backward_hooks\n\u001b[1;32m   1540\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_forward_pre_hooks):\n\u001b[0;32m-> 1541\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mforward_call\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1543\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m   1544\u001b[0m     result \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/activation.py:103\u001b[0m, in \u001b[0;36mReLU.forward\u001b[0;34m(self, input)\u001b[0m\n\u001b[1;32m    102\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mforward\u001b[39m(\u001b[38;5;28mself\u001b[39m, \u001b[38;5;28minput\u001b[39m: Tensor) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m Tensor:\n\u001b[0;32m--> 103\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mF\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mrelu\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43minput\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43minplace\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43minplace\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/functional.py:1500\u001b[0m, in \u001b[0;36mrelu\u001b[0;34m(input, inplace)\u001b[0m\n\u001b[1;32m   1498\u001b[0m     result \u001b[38;5;241m=\u001b[39m torch\u001b[38;5;241m.\u001b[39mrelu_(\u001b[38;5;28minput\u001b[39m)\n\u001b[1;32m   1499\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[0;32m-> 1500\u001b[0m     result \u001b[38;5;241m=\u001b[39m \u001b[43mtorch\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mrelu\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43minput\u001b[39;49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1501\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m result\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "###################################### Clustered FL training \n",
    "# ============================================================\n",
    "# Helpers\n",
    "# ============================================================\n",
    "\n",
    "from collections import OrderedDict\n",
    "\n",
    "def cluster_participants(idxs_users, clients_clust_id):\n",
    "    \"\"\"\n",
    "    Map sampled user IDs to {cluster_id: [user_ids]}.\n",
    "    \"\"\"\n",
    "    out = {}\n",
    "    for uid in idxs_users:\n",
    "        cid = clients_clust_id[uid]\n",
    "        out.setdefault(cid, []).append(uid)\n",
    "    return out\n",
    "\n",
    "\n",
    "def make_weight_vec(id_list, net_dataidx_map):\n",
    "    \"\"\"\n",
    "    Return FedAvg weights proportional to each client's data size.\n",
    "    \"\"\"\n",
    "    sizes = [len(net_dataidx_map[u]) for u in id_list]\n",
    "    tot   = sum(sizes)\n",
    "    return [s / tot for s in sizes]\n",
    "\n",
    "\n",
    "def FedAvg_trainable(sd_list, weights):\n",
    "    \"\"\"\n",
    "    FedAvg only the trainable blocks: own_encoder.* and classifier.*.\n",
    "    Frozen secondary_* keys are left untouched.\n",
    "    \"\"\"\n",
    "    out = copy.deepcopy(sd_list[0])\n",
    "    with torch.no_grad():\n",
    "        for k in out:\n",
    "            if k.startswith((\"own_encoder\", \"classifier\")):\n",
    "                out[k].zero_()\n",
    "                for w, sd in zip(weights, sd_list):\n",
    "                    out[k] += w * sd[k]\n",
    "    return out\n",
    "\n",
    "\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",
    "\n",
    "n_rounds   = 41\n",
    "print_step = 10              # stats cadence after warm-up\n",
    "loss_buf   = []\n",
    "\n",
    "for rnd in range(n_rounds):\n",
    "\n",
    "    print(f\"\\n##### ROUND {rnd+1} #####\")\n",
    "\n",
    "    # 1) sample users\n",
    "    m            = int(args.frac * args.num_users)\n",
    "    idxs_users   = np.random.choice(args.num_users, m, replace=False)\n",
    "    clusters_now = cluster_participants(idxs_users, clients_clust_id)\n",
    "\n",
    "    # 2) broadcast combined cluster model to each selected user\n",
    "    for uid in idxs_users:\n",
    "        cid = clients_clust_id[uid]\n",
    "        clients[uid].set_state_dict(copy.deepcopy(w_glob_per_cluster[cid]))\n",
    "\n",
    "    # 3) per-cluster FedAvg containers\n",
    "    enc_pool, clf_pool, freq_pool = {}, {}, {}\n",
    "\n",
    "    # 4) local training\n",
    "    for uid in idxs_users:\n",
    "        cid   = clients_clust_id[uid]\n",
    "        ndata = len(net_dataidx_map[uid])\n",
    "\n",
    "        # ---- metrics before train\n",
    "        pre_loss, pre_acc = clients[uid].eval_test()\n",
    "        client_metric[uid][\"best_pre\"]  = max(client_metric[uid][\"best_pre\"],  pre_acc)\n",
    "\n",
    "        # ---- local SGD (own_encoder + classifier only)\n",
    "        loss = clients[uid].train2()\n",
    "        loss_buf.append(loss)\n",
    "\n",
    "        # ---- metrics after train\n",
    "        post_loss, post_acc = clients[uid].eval_test()\n",
    "        client_metric[uid][\"best_post\"] = max(client_metric[uid][\"best_post\"], post_acc)\n",
    "        client_metric[uid][\"best_any\"]  = max(client_metric[uid][\"best_any\"],\n",
    "                                              pre_acc, post_acc)\n",
    "\n",
    "        # ---- stash trainable weights for FedAvg\n",
    "        # Wrap encoder weights with prefix\n",
    "        own_sd = clients[uid].net.own_encoder.state_dict()\n",
    "        own_sd_prefixed = OrderedDict({f\"own_encoder.{k}\": v for k, v in own_sd.items()})\n",
    "        enc_pool.setdefault(cid, []).append(own_sd_prefixed)\n",
    "        \n",
    "        # Wrap classifier weights with prefix\n",
    "        clf_sd = clients[uid].net.classifier.state_dict()\n",
    "        clf_sd_prefixed = OrderedDict({f\"classifier.{k}\": v for k, v in clf_sd.items()})\n",
    "        clf_pool.setdefault(cid, []).append(clf_sd_prefixed)\n",
    "\n",
    "    # 5) FedAvg per cluster (trainable blocks)\n",
    "    for cid in enc_pool:\n",
    "        weights = make_weight_vec(clusters_now[cid], net_dataidx_map)\n",
    "        upd_enc = FedAvg_trainable(enc_pool[cid], weights)\n",
    "        upd_clf = FedAvg_trainable(clf_pool[cid], weights)\n",
    "        w_glob_per_cluster[cid].update(upd_enc)\n",
    "        w_glob_per_cluster[cid].update(upd_clf)\n",
    "\n",
    "    # 6) periodic statistics\n",
    "    if rnd < 4:\n",
    "        print_step = 1\n",
    "    else:\n",
    "        print_step = 10\n",
    "\n",
    "    if rnd % print_step == 0:\n",
    "        avg_loss = np.mean(loss_buf) if loss_buf else 0.0\n",
    "        avg_pre  = np.mean([m[\"best_pre\"]  for m in client_metric.values()])\n",
    "        avg_post = np.mean([m[\"best_post\"] for m in client_metric.values()])\n",
    "        avg_any  = np.mean([m[\"best_any\"]  for m in client_metric.values()])\n",
    "\n",
    "        print(\"---- ROUND STATS ----\")\n",
    "        print(f\"avg train loss : {avg_loss:.4f}\")\n",
    "        print(f\"avg best PRE   : {avg_pre :.2f}%\")\n",
    "        print(f\"avg best POST  : {avg_post:.2f}%\")\n",
    "        print(f\"avg best ANY   : {avg_any :.2f}%\\n\")\n",
    "\n",
    "        # optional per-client line (comment if too verbose)\n",
    "        for uid in range(args.num_users):\n",
    "            bm = client_metric[uid]\n",
    "            print(f\"Client {uid:3d} | best_pre={bm['best_pre'] :6.2f} \"\n",
    "                  f\"| best_post={bm['best_post']:6.2f} \"\n",
    "                  f\"| best_any={bm['best_any'] :6.2f}\")\n",
    "\n",
    "        # reset buffers for next report window\n",
    "        loss_buf.clear()\n",
    "        gc.collect()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "print(\"hello\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{0: [1, 2], 1: [4, 0], 2: [1, 4], 3: [4, 0], 4: [2, 1]}\n",
      "Cluster 0, First Client 0: {0: 433, 2: 191, 9: 238}\n",
      "Cluster 1, First Client 1: {2: 616, 6: 115, 8: 18}\n",
      "Cluster 2, First Client 2: {1: 693, 7: 45, 8: 59}\n",
      "Cluster 3, First Client 8: {3: 632, 4: 17, 5: 41}\n",
      "Cluster 4, First Client 10: {4: 265, 6: 215, 7: 240}\n"
     ]
    }
   ],
   "source": [
    "\n",
    "import numpy as np\n",
    "from typing import Dict, List\n",
    "\n",
    "def build_similarity_graph_rank_supply_nonzero(\n",
    "        traindata_cls_counts: Dict[int, Dict[int, int]],\n",
    "        clusters: List[List[int]],\n",
    "        num_classes: int = 10,\n",
    "        top_k: int = 2\n",
    ") -> Dict[int, List[int]]:\n",
    "    \"\"\"\n",
    "    Build a directed Cluster-Complementarity Graph.\n",
    "    Demand is high for rare classes; supply is high for abundant classes.\n",
    "\n",
    "    Parameters\n",
    "    ----------\n",
    "    traindata_cls_counts : {client_id: {class: count}}\n",
    "    clusters             : list of clusters (each is a list of client IDs)\n",
    "    num_classes          : total number of classes\n",
    "    top_k                : keep this many outgoing edges per cluster\n",
    "\n",
    "    Returns\n",
    "    -------\n",
    "    similarity_graph : {cluster_id: [target clusters]}\n",
    "    \"\"\"\n",
    "    C, K = len(clusters), num_classes\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 1. Per-client rank over *non-zero* classes\n",
    "    # ----------------------------------------------------------\n",
    "    client_rank = [{} for _ in range(len(traindata_cls_counts))]\n",
    "    for cid, counts in traindata_cls_counts.items():\n",
    "        present = [(cls, cnt) for cls, cnt in counts.items() if cnt > 0]\n",
    "        if not present:\n",
    "            continue                                # client has no data\n",
    "        present.sort(key=lambda x: x[1])            # ascending (rarest first)\n",
    "        for r, (cls, _) in enumerate(present):\n",
    "            client_rank[cid][cls] = r               # rank 0 … m_i-1\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 2. Demand weight  w[p,k]   (large if class k is rare in cluster p)\n",
    "    # ----------------------------------------------------------\n",
    "    w = np.zeros((C, K), dtype=float)\n",
    "    for p, clist in enumerate(clusters):\n",
    "        for cid in clist:\n",
    "            m_i = len(client_rank[cid])             # number of classes present\n",
    "            for cls, r in client_rank[cid].items():\n",
    "                w[p, cls] += (m_i - r)              # larger when rarer\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 3. Supply strength s[q,k] (large if class k is abundant in cluster q)\n",
    "    # ----------------------------------------------------------\n",
    "    s = np.zeros((C, K), dtype=float)\n",
    "    for q, clist in enumerate(clusters):\n",
    "        for cid in clist:\n",
    "            for cls, r in client_rank[cid].items():\n",
    "                s[q, cls] += (r + 1)                # larger when more common\n",
    "        n_q = max(len(clist), 1)\n",
    "        s[q] /= n_q                                 # per-client normalisation\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 4. Complementarity score matrix  A = W · Sᵀ\n",
    "    # ----------------------------------------------------------\n",
    "    score = w @ s.T\n",
    "    np.fill_diagonal(score, -np.inf)                # forbid self-loops\n",
    "\n",
    "    # ----------------------------------------------------------\n",
    "    # 5. Keep top-k outgoing edges for each cluster\n",
    "    # ----------------------------------------------------------\n",
    "    graph = {p: np.argsort(-score[p])[:top_k].tolist() for p in range(C)}\n",
    "    return graph\n",
    "\n",
    "\n",
    "    \n",
    "similarity_graph = build_similarity_graph_rank_supply_nonzero(\n",
    "        traindata_cls_counts,   # your dict\n",
    "        clusters,               # your list of clusters\n",
    "        num_classes=10,\n",
    "        top_k=2\n",
    ")\n",
    "print(similarity_graph)\n",
    "\n",
    "for k in range(len(clusters)):\n",
    "    print(f\"Cluster {k}, First Client {clusters[k][0]}:\", traindata_cls_counts[clusters[k][0]])\n",
    "     "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "metadata": {},
   "outputs": [],
   "source": [
    "def to_model(sd):\n",
    "    mdl = SimpleCNNMNIST2()          # same arch #edit needed, pass args and change based on data\n",
    "    mdl.load_state_dict(sd)\n",
    "    return mdl\n",
    "\n",
    "\n",
    "def FedAvg2(state_dict_list, weight_avg):\n",
    "    \"\"\"\n",
    "    Average a *list* of state_dicts; weight_avg[i] is that model’s weight.\n",
    "    Returns a fresh averaged state_dict.\n",
    "    \"\"\"\n",
    "    averaged = copy.deepcopy(state_dict_list[0])\n",
    "    for key in averaged.keys():\n",
    "        averaged[key] = sum(w * sd[key] for w, sd in zip(weight_avg, state_dict_list))\n",
    "    return averaged\n",
    "\n",
    "def build_secondary_encoder(target_cluster_id: int,\n",
    "                            glob_state_dicts,        # <-- list of OrderedDict\n",
    "                            similarity_graph: dict):\n",
    "    # create target model just to grab its encoder structure\n",
    "    sec_enc = copy.deepcopy(to_model(glob_state_dicts[target_cluster_id]).encoder)\n",
    "    for p in sec_enc.parameters():\n",
    "        p.data.zero_()\n",
    "\n",
    "    # accumulate encoders from similar clusters\n",
    "    for cid in similarity_graph.get(target_cluster_id, []):\n",
    "        enc = to_model(glob_state_dicts[cid]).encoder\n",
    "        for sec_p, src_p in zip(sec_enc.parameters(), enc.parameters()):\n",
    "            sec_p.data += src_p.data\n",
    "    return sec_enc\n",
    "\n",
    "def build_secondary_classifier(\n",
    "        target_cluster_id: int,\n",
    "        glob_state_dicts: List[Dict],        # Phase-1 state-dicts\n",
    "        similarity_graph: Dict[int, List[int]]\n",
    "    ) -> Dict:\n",
    "    \"\"\"\n",
    "    Returns an averaged classifier state_dict (weight & bias) of the\n",
    "    neighbour clusters connected to `target_cluster_id`.\n",
    "    \"\"\"\n",
    "    neighbours = similarity_graph.get(target_cluster_id, [])\n",
    "    if not neighbours:                       # no neighbours ⇒ zero init\n",
    "        # weight: (10,84) ; bias: (10,)\n",
    "        return {\n",
    "            'weight': torch.zeros_like(\n",
    "                glob_state_dicts[target_cluster_id]['classifier.weight']),\n",
    "            'bias':   torch.zeros_like(\n",
    "                glob_state_dicts[target_cluster_id]['classifier.bias']),\n",
    "        }\n",
    "\n",
    "    # ---- accumulate weights / biases ----\n",
    "    w_sum = 0.0\n",
    "    b_sum = 0.0\n",
    "    for cid in neighbours:\n",
    "        w_sum += glob_state_dicts[cid]['classifier.weight']\n",
    "        b_sum += glob_state_dicts[cid]['classifier.bias']\n",
    "    w_avg = w_sum / len(neighbours)\n",
    "    b_avg = b_sum / len(neighbours)\n",
    "\n",
    "    return {'weight': w_avg.clone(), 'bias': b_avg.clone()}\n",
    "\n",
    "# --- Assume you already ran Phase-1 & have: ----------------------------\n",
    "# clusters               : List[List[int]] – client indices per cluster\n",
    "# global_cluster_models  : Dict[int, SimpleCNN] – best model per cluster\n",
    "# similarity_graph       : Dict[int, List[int]] – your similarity edges\n",
    "# clients                : List[Client_ClusterFL] (place-holders for now)\n",
    "# -----------------------------------------------------------------------\n",
    "\n",
    "# ------ Build CombinedModel and push to each client  -------------------\n",
    "### >>> MODIFIED LOOP STARTS HERE <<<\n",
    "global_cluster_models = copy.deepcopy(w_glob_per_cluster)   # Phase-1 best SDs\n",
    "\n",
    "for cid, client_idxs in enumerate(clusters):\n",
    "    # ---- rebuild Phase-1 model once -------------------------\n",
    "    own_model = to_model(global_cluster_models[cid])         # SimpleCNNMNIST2()\n",
    "    own_enc   = own_model.encoder\n",
    "    own_clf_sd = own_model.classifier.state_dict()\n",
    "\n",
    "    # ---- secondary encoder ---------------------------------\n",
    "    sec_enc = build_secondary_encoder(\n",
    "        cid,\n",
    "        global_cluster_models,\n",
    "        similarity_graph\n",
    "    )\n",
    "\n",
    "    # ---- secondary classifier ------------------------------\n",
    "    sec_clf_sd = build_secondary_classifier(\n",
    "        cid,\n",
    "        global_cluster_models,\n",
    "        similarity_graph\n",
    "    )\n",
    "\n",
    "    # ---- assign CombinedModel to every client --------------\n",
    "    for idx in client_idxs:\n",
    "        clients[idx].net = CombinedModelMNIST(\n",
    "            own_encoder       = own_enc,\n",
    "            secondary_encoder = sec_enc,\n",
    "            own_clf_sd        = own_clf_sd,\n",
    "            sec_clf_sd        = sec_clf_sd,\n",
    "            num_classes       = 10\n",
    "        )\n",
    "### <<< MODIFIED LOOP ENDS HERE >>>"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "###### P2-ROUND 1 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0192\n",
      "Init Test Acc:  76.23 Final Test Acc: 94.18\n",
      "Client   0  cur=92.17  best=92.17\n",
      "Client   1  cur=52.00  best= 0.00\n",
      "Client   2  cur=93.80  best=93.80\n",
      "Client   3  cur=52.00  best= 0.00\n",
      "Client   4  cur=85.60  best= 0.00\n",
      "Client   5  cur=95.70  best=95.70\n",
      "Client   6  cur=82.23  best= 0.00\n",
      "Client   7  cur=85.60  best= 0.00\n",
      "Client   8  cur=85.60  best= 0.00\n",
      "Client   9  cur=62.27  best= 0.00\n",
      "Client  10  cur=62.27  best= 0.00\n",
      "Client  11  cur=82.23  best= 0.00\n",
      "Client  12  cur=52.00  best= 0.00\n",
      "Client  13  cur=85.60  best= 0.00\n",
      "Client  14  cur=62.27  best= 0.00\n",
      "Client  15  cur=85.67  best= 0.00\n",
      "Client  16  cur=62.27  best= 0.00\n",
      "Client  17  cur=85.60  best= 0.00\n",
      "Client  18  cur=82.23  best= 0.00\n",
      "Client  19  cur=52.00  best= 0.00\n",
      "Client  20  cur=62.27  best= 0.00\n",
      "Client  21  cur=82.23  best= 0.00\n",
      "Client  22  cur=52.00  best= 0.00\n",
      "Client  23  cur=85.67  best= 0.00\n",
      "Client  24  cur=85.60  best= 0.00\n",
      "Client  25  cur=95.67  best=95.67\n",
      "Client  26  cur=52.00  best= 0.00\n",
      "Client  27  cur=62.27  best= 0.00\n",
      "Client  28  cur=82.23  best= 0.00\n",
      "Client  29  cur=96.87  best=96.87\n",
      "Client  30  cur=85.67  best= 0.00\n",
      "Client  31  cur=85.67  best= 0.00\n",
      "Client  32  cur=82.23  best= 0.00\n",
      "Client  33  cur=95.20  best=95.20\n",
      "Client  34  cur=52.00  best= 0.00\n",
      "Client  35  cur=98.13  best=98.13\n",
      "Client  36  cur=82.23  best= 0.00\n",
      "Client  37  cur=82.23  best= 0.00\n",
      "Client  38  cur=85.60  best= 0.00\n",
      "Client  39  cur=62.27  best= 0.00\n",
      "Client  40  cur=52.00  best= 0.00\n",
      "Client  41  cur=62.27  best= 0.00\n",
      "Client  42  cur=62.27  best= 0.00\n",
      "Client  43  cur=62.27  best= 0.00\n",
      "Client  44  cur=97.50  best=97.50\n",
      "Client  45  cur=52.00  best= 0.00\n",
      "Client  46  cur=91.57  best=91.57\n",
      "Client  47  cur=85.60  best= 0.00\n",
      "Client  48  cur=62.27  best= 0.00\n",
      "Client  49  cur=52.00  best= 0.00\n",
      "Client  50  cur=95.90  best=95.90\n",
      "Client  51  cur=85.67  best= 0.00\n",
      "Client  52  cur=85.67  best= 0.00\n",
      "Client  53  cur=72.73  best=72.73\n",
      "Client  54  cur=94.33  best=94.33\n",
      "Client  55  cur=62.27  best= 0.00\n",
      "Client  56  cur=82.23  best= 0.00\n",
      "Client  57  cur=82.23  best= 0.00\n",
      "Client  58  cur=85.60  best= 0.00\n",
      "Client  59  cur=62.27  best= 0.00\n",
      "Client  60  cur=85.67  best= 0.00\n",
      "Client  61  cur=99.07  best=99.07\n",
      "Client  62  cur=82.23  best= 0.00\n",
      "Client  63  cur=52.00  best= 0.00\n",
      "Client  64  cur=95.33  best=95.33\n",
      "Client  65  cur=85.67  best= 0.00\n",
      "Client  66  cur=85.60  best= 0.00\n",
      "Client  67  cur=96.57  best=96.57\n",
      "Client  68  cur=85.60  best= 0.00\n",
      "Client  69  cur=62.27  best= 0.00\n",
      "Client  70  cur=82.23  best= 0.00\n",
      "Client  71  cur=62.27  best= 0.00\n",
      "Client  72  cur=52.00  best= 0.00\n",
      "Client  73  cur=98.67  best=98.67\n",
      "Client  74  cur=85.67  best= 0.00\n",
      "Client  75  cur=62.27  best= 0.00\n",
      "Client  76  cur=85.67  best= 0.00\n",
      "Client  77  cur=82.23  best= 0.00\n",
      "Client  78  cur=62.27  best= 0.00\n",
      "Client  79  cur=98.83  best=98.83\n",
      "Client  80  cur=90.70  best=90.70\n",
      "Client  81  cur=90.87  best=90.87\n",
      "Client  82  cur=85.60  best= 0.00\n",
      "Client  83  cur=85.60  best= 0.00\n",
      "Client  84  cur=85.67  best= 0.00\n",
      "Client  85  cur=62.27  best= 0.00\n",
      "Client  86  cur=85.67  best= 0.00\n",
      "Client  87  cur=85.60  best= 0.00\n",
      "Client  88  cur=62.27  best= 0.00\n",
      "Client  89  cur=85.67  best= 0.00\n",
      "Client  90  cur=85.60  best= 0.00\n",
      "Client  91  cur=52.00  best= 0.00\n",
      "Client  92  cur=82.23  best= 0.00\n",
      "Client  93  cur=52.00  best= 0.00\n",
      "Client  94  cur=85.60  best= 0.00\n",
      "Client  95  cur=85.60  best= 0.00\n",
      "Client  96  cur=85.60  best= 0.00\n",
      "Client  97  cur=85.60  best= 0.00\n",
      "Client  98  cur=94.07  best=94.07\n",
      "Client  99  cur=82.23  best= 0.00\n",
      "Round 1 AvgCur=77.68  AvgBest=18.84\n",
      "\n",
      "###### P2-ROUND 2 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0159\n",
      "Init Test Acc:  96.38 Final Test Acc: 95.35\n",
      "Client   0  cur=92.17  best=92.17\n",
      "Client   1  cur=52.00  best= 0.00\n",
      "Client   2  cur=93.80  best=93.80\n",
      "Client   3  cur=98.43  best=98.97\n",
      "Client   4  cur=85.60  best= 0.00\n",
      "Client   5  cur=95.70  best=95.70\n",
      "Client   6  cur=82.23  best= 0.00\n",
      "Client   7  cur=85.60  best= 0.00\n",
      "Client   8  cur=95.27  best=95.83\n",
      "Client   9  cur=62.27  best= 0.00\n",
      "Client  10  cur=62.27  best= 0.00\n",
      "Client  11  cur=82.23  best= 0.00\n",
      "Client  12  cur=97.73  best=98.97\n",
      "Client  13  cur=93.30  best=95.83\n",
      "Client  14  cur=62.27  best= 0.00\n",
      "Client  15  cur=93.17  best=96.00\n",
      "Client  16  cur=62.27  best= 0.00\n",
      "Client  17  cur=94.93  best=95.83\n",
      "Client  18  cur=82.23  best= 0.00\n",
      "Client  19  cur=98.97  best=98.97\n",
      "Client  20  cur=62.27  best= 0.00\n",
      "Client  21  cur=82.23  best= 0.00\n",
      "Client  22  cur=52.00  best= 0.00\n",
      "Client  23  cur=85.67  best= 0.00\n",
      "Client  24  cur=85.60  best= 0.00\n",
      "Client  25  cur=95.67  best=95.67\n",
      "Client  26  cur=52.00  best= 0.00\n",
      "Client  27  cur=62.27  best= 0.00\n",
      "Client  28  cur=82.23  best= 0.00\n",
      "Client  29  cur=96.87  best=96.87\n",
      "Client  30  cur=95.87  best=96.00\n",
      "Client  31  cur=85.67  best= 0.00\n",
      "Client  32  cur=82.23  best= 0.00\n",
      "Client  33  cur=95.20  best=95.20\n",
      "Client  34  cur=98.97  best=98.97\n",
      "Client  35  cur=98.13  best=98.13\n",
      "Client  36  cur=98.47  best=98.47\n",
      "Client  37  cur=97.30  best=98.23\n",
      "Client  38  cur=85.60  best= 0.00\n",
      "Client  39  cur=62.27  best= 0.00\n",
      "Client  40  cur=52.00  best= 0.00\n",
      "Client  41  cur=92.13  best=92.13\n",
      "Client  42  cur=62.27  best= 0.00\n",
      "Client  43  cur=92.87  best=92.87\n",
      "Client  44  cur=97.50  best=97.50\n",
      "Client  45  cur=52.00  best= 0.00\n",
      "Client  46  cur=91.57  best=91.57\n",
      "Client  47  cur=85.60  best= 0.00\n",
      "Client  48  cur=91.67  best=91.70\n",
      "Client  49  cur=52.00  best= 0.00\n",
      "Client  50  cur=95.90  best=95.90\n",
      "Client  51  cur=85.67  best= 0.00\n",
      "Client  52  cur=85.67  best= 0.00\n",
      "Client  53  cur=72.73  best=72.73\n",
      "Client  54  cur=93.37  best=95.83\n",
      "Client  55  cur=62.27  best= 0.00\n",
      "Client  56  cur=82.23  best= 0.00\n",
      "Client  57  cur=98.03  best=98.23\n",
      "Client  58  cur=85.60  best= 0.00\n",
      "Client  59  cur=62.27  best= 0.00\n",
      "Client  60  cur=94.83  best=96.00\n",
      "Client  61  cur=99.07  best=99.07\n",
      "Client  62  cur=82.23  best= 0.00\n",
      "Client  63  cur=52.00  best= 0.00\n",
      "Client  64  cur=95.33  best=95.33\n",
      "Client  65  cur=85.67  best= 0.00\n",
      "Client  66  cur=85.60  best= 0.00\n",
      "Client  67  cur=96.57  best=96.57\n",
      "Client  68  cur=85.60  best= 0.00\n",
      "Client  69  cur=62.27  best= 0.00\n",
      "Client  70  cur=82.23  best= 0.00\n",
      "Client  71  cur=62.27  best= 0.00\n",
      "Client  72  cur=52.00  best= 0.00\n",
      "Client  73  cur=98.67  best=98.67\n",
      "Client  74  cur=85.67  best= 0.00\n",
      "Client  75  cur=62.27  best= 0.00\n",
      "Client  76  cur=85.67  best= 0.00\n",
      "Client  77  cur=82.23  best= 0.00\n",
      "Client  78  cur=62.27  best= 0.00\n",
      "Client  79  cur=98.97  best=98.97\n",
      "Client  80  cur=90.70  best=90.70\n",
      "Client  81  cur=90.87  best=90.87\n",
      "Client  82  cur=85.60  best= 0.00\n",
      "Client  83  cur=85.60  best= 0.00\n",
      "Client  84  cur=85.67  best= 0.00\n",
      "Client  85  cur=62.27  best= 0.00\n",
      "Client  86  cur=85.67  best= 0.00\n",
      "Client  87  cur=95.73  best=95.83\n",
      "Client  88  cur=62.27  best= 0.00\n",
      "Client  89  cur=85.67  best= 0.00\n",
      "Client  90  cur=85.60  best= 0.00\n",
      "Client  91  cur=52.00  best= 0.00\n",
      "Client  92  cur=82.23  best= 0.00\n",
      "Client  93  cur=52.00  best= 0.00\n",
      "Client  94  cur=85.60  best= 0.00\n",
      "Client  95  cur=85.60  best= 0.00\n",
      "Client  96  cur=87.07  best=95.83\n",
      "Client  97  cur=85.60  best= 0.00\n",
      "Client  98  cur=94.07  best=94.07\n",
      "Client  99  cur=82.23  best= 0.00\n",
      "Round 2 AvgCur=81.56  AvgBest=36.20\n",
      "\n",
      "###### P2-ROUND 3 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0140\n",
      "Init Test Acc:  96.17 Final Test Acc: 94.87\n",
      "Client   0  cur=93.30  best=95.77\n",
      "Client   1  cur=52.00  best= 0.00\n",
      "Client   2  cur=93.80  best=93.80\n",
      "Client   3  cur=98.43  best=98.97\n",
      "Client   4  cur=85.60  best= 0.00\n",
      "Client   5  cur=95.70  best=95.70\n",
      "Client   6  cur=82.23  best= 0.00\n",
      "Client   7  cur=95.53  best=95.77\n",
      "Client   8  cur=95.27  best=95.83\n",
      "Client   9  cur=62.27  best= 0.00\n",
      "Client  10  cur=62.27  best= 0.00\n",
      "Client  11  cur=82.23  best= 0.00\n",
      "Client  12  cur=97.73  best=98.97\n",
      "Client  13  cur=93.30  best=95.83\n",
      "Client  14  cur=62.27  best= 0.00\n",
      "Client  15  cur=93.17  best=96.00\n",
      "Client  16  cur=92.70  best=93.00\n",
      "Client  17  cur=94.93  best=95.83\n",
      "Client  18  cur=82.23  best= 0.00\n",
      "Client  19  cur=98.97  best=98.97\n",
      "Client  20  cur=62.27  best= 0.00\n",
      "Client  21  cur=82.23  best= 0.00\n",
      "Client  22  cur=52.00  best= 0.00\n",
      "Client  23  cur=85.67  best= 0.00\n",
      "Client  24  cur=94.33  best=95.77\n",
      "Client  25  cur=95.67  best=95.67\n",
      "Client  26  cur=52.00  best= 0.00\n",
      "Client  27  cur=62.27  best= 0.00\n",
      "Client  28  cur=82.23  best= 0.00\n",
      "Client  29  cur=96.87  best=97.73\n",
      "Client  30  cur=95.87  best=96.00\n",
      "Client  31  cur=85.67  best= 0.00\n",
      "Client  32  cur=82.23  best= 0.00\n",
      "Client  33  cur=95.47  best=95.47\n",
      "Client  34  cur=98.97  best=98.97\n",
      "Client  35  cur=98.13  best=98.13\n",
      "Client  36  cur=98.47  best=98.47\n",
      "Client  37  cur=97.23  best=98.23\n",
      "Client  38  cur=85.60  best= 0.00\n",
      "Client  39  cur=62.27  best= 0.00\n",
      "Client  40  cur=52.00  best= 0.00\n",
      "Client  41  cur=92.13  best=92.13\n",
      "Client  42  cur=62.27  best= 0.00\n",
      "Client  43  cur=92.43  best=93.00\n",
      "Client  44  cur=97.50  best=97.50\n",
      "Client  45  cur=52.00  best= 0.00\n",
      "Client  46  cur=91.63  best=95.77\n",
      "Client  47  cur=85.60  best= 0.00\n",
      "Client  48  cur=91.67  best=91.70\n",
      "Client  49  cur=52.00  best= 0.00\n",
      "Client  50  cur=95.90  best=95.90\n",
      "Client  51  cur=95.10  best=95.33\n",
      "Client  52  cur=85.67  best= 0.00\n",
      "Client  53  cur=76.57  best=93.00\n",
      "Client  54  cur=93.37  best=95.83\n",
      "Client  55  cur=62.27  best= 0.00\n",
      "Client  56  cur=98.07  best=98.07\n",
      "Client  57  cur=98.03  best=98.23\n",
      "Client  58  cur=85.60  best= 0.00\n",
      "Client  59  cur=92.67  best=93.00\n",
      "Client  60  cur=94.83  best=96.00\n",
      "Client  61  cur=99.03  best=99.07\n",
      "Client  62  cur=82.23  best= 0.00\n",
      "Client  63  cur=52.00  best= 0.00\n",
      "Client  64  cur=95.70  best=95.70\n",
      "Client  65  cur=85.67  best= 0.00\n",
      "Client  66  cur=85.60  best= 0.00\n",
      "Client  67  cur=96.57  best=96.57\n",
      "Client  68  cur=85.60  best= 0.00\n",
      "Client  69  cur=62.27  best= 0.00\n",
      "Client  70  cur=82.23  best= 0.00\n",
      "Client  71  cur=62.27  best= 0.00\n",
      "Client  72  cur=99.10  best=99.10\n",
      "Client  73  cur=98.90  best=99.00\n",
      "Client  74  cur=96.33  best=96.33\n",
      "Client  75  cur=62.27  best= 0.00\n",
      "Client  76  cur=85.67  best= 0.00\n",
      "Client  77  cur=82.23  best= 0.00\n",
      "Client  78  cur=62.27  best= 0.00\n",
      "Client  79  cur=99.00  best=99.00\n",
      "Client  80  cur=90.70  best=90.70\n",
      "Client  81  cur=90.87  best=90.87\n",
      "Client  82  cur=85.60  best= 0.00\n",
      "Client  83  cur=85.60  best= 0.00\n",
      "Client  84  cur=85.67  best= 0.00\n",
      "Client  85  cur=62.27  best= 0.00\n",
      "Client  86  cur=85.67  best= 0.00\n",
      "Client  87  cur=95.73  best=95.83\n",
      "Client  88  cur=62.27  best= 0.00\n",
      "Client  89  cur=85.67  best= 0.00\n",
      "Client  90  cur=85.60  best= 0.00\n",
      "Client  91  cur=52.00  best= 0.00\n",
      "Client  92  cur=82.23  best= 0.00\n",
      "Client  93  cur=52.00  best= 0.00\n",
      "Client  94  cur=85.60  best= 0.00\n",
      "Client  95  cur=85.60  best= 0.00\n",
      "Client  96  cur=87.07  best=95.83\n",
      "Client  97  cur=85.60  best= 0.00\n",
      "Client  98  cur=94.07  best=94.07\n",
      "Client  99  cur=97.47  best=97.73\n",
      "Round 3 AvgCur=83.39  AvgBest=45.14\n",
      "\n",
      "###### P2-ROUND 4 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0154\n",
      "Init Test Acc:  95.78 Final Test Acc: 94.26\n",
      "Client   0  cur=93.30  best=95.77\n",
      "Client   1  cur=52.00  best= 0.00\n",
      "Client   2  cur=93.80  best=93.80\n",
      "Client   3  cur=98.43  best=98.97\n",
      "Client   4  cur=85.60  best= 0.00\n",
      "Client   5  cur=95.70  best=95.70\n",
      "Client   6  cur=82.23  best= 0.00\n",
      "Client   7  cur=95.53  best=95.77\n",
      "Client   8  cur=95.27  best=95.83\n",
      "Client   9  cur=62.27  best= 0.00\n",
      "Client  10  cur=62.27  best= 0.00\n",
      "Client  11  cur=82.23  best= 0.00\n",
      "Client  12  cur=98.53  best=99.00\n",
      "Client  13  cur=93.30  best=95.83\n",
      "Client  14  cur=62.27  best= 0.00\n",
      "Client  15  cur=93.17  best=96.00\n",
      "Client  16  cur=92.70  best=93.00\n",
      "Client  17  cur=94.93  best=95.83\n",
      "Client  18  cur=98.50  best=98.50\n",
      "Client  19  cur=98.97  best=98.97\n",
      "Client  20  cur=92.07  best=92.53\n",
      "Client  21  cur=97.83  best=97.87\n",
      "Client  22  cur=52.00  best= 0.00\n",
      "Client  23  cur=85.67  best= 0.00\n",
      "Client  24  cur=94.43  best=96.10\n",
      "Client  25  cur=95.67  best=95.67\n",
      "Client  26  cur=52.00  best= 0.00\n",
      "Client  27  cur=62.27  best= 0.00\n",
      "Client  28  cur=82.23  best= 0.00\n",
      "Client  29  cur=96.77  best=97.87\n",
      "Client  30  cur=95.87  best=96.00\n",
      "Client  31  cur=85.67  best= 0.00\n",
      "Client  32  cur=82.23  best= 0.00\n",
      "Client  33  cur=95.47  best=95.47\n",
      "Client  34  cur=98.97  best=98.97\n",
      "Client  35  cur=98.13  best=98.13\n",
      "Client  36  cur=98.33  best=98.47\n",
      "Client  37  cur=97.23  best=98.23\n",
      "Client  38  cur=85.60  best= 0.00\n",
      "Client  39  cur=92.60  best=92.60\n",
      "Client  40  cur=52.00  best= 0.00\n",
      "Client  41  cur=92.13  best=92.13\n",
      "Client  42  cur=86.13  best=92.53\n",
      "Client  43  cur=92.43  best=93.00\n",
      "Client  44  cur=97.50  best=97.50\n",
      "Client  45  cur=52.00  best= 0.00\n",
      "Client  46  cur=91.63  best=95.77\n",
      "Client  47  cur=87.57  best=96.10\n",
      "Client  48  cur=91.67  best=91.70\n",
      "Client  49  cur=52.00  best= 0.00\n",
      "Client  50  cur=95.90  best=95.90\n",
      "Client  51  cur=94.97  best=96.60\n",
      "Client  52  cur=95.23  best=96.60\n",
      "Client  53  cur=76.57  best=93.00\n",
      "Client  54  cur=93.37  best=95.83\n",
      "Client  55  cur=62.27  best= 0.00\n",
      "Client  56  cur=98.07  best=98.07\n",
      "Client  57  cur=98.03  best=98.23\n",
      "Client  58  cur=85.60  best= 0.00\n",
      "Client  59  cur=92.67  best=93.00\n",
      "Client  60  cur=94.33  best=96.60\n",
      "Client  61  cur=99.03  best=99.07\n",
      "Client  62  cur=82.23  best= 0.00\n",
      "Client  63  cur=52.00  best= 0.00\n",
      "Client  64  cur=95.70  best=95.70\n",
      "Client  65  cur=85.67  best= 0.00\n",
      "Client  66  cur=85.60  best= 0.00\n",
      "Client  67  cur=96.90  best=97.87\n",
      "Client  68  cur=95.70  best=96.10\n",
      "Client  69  cur=62.27  best= 0.00\n",
      "Client  70  cur=82.23  best= 0.00\n",
      "Client  71  cur=62.27  best= 0.00\n",
      "Client  72  cur=99.10  best=99.10\n",
      "Client  73  cur=98.90  best=99.00\n",
      "Client  74  cur=96.33  best=96.33\n",
      "Client  75  cur=62.27  best= 0.00\n",
      "Client  76  cur=85.67  best= 0.00\n",
      "Client  77  cur=82.23  best= 0.00\n",
      "Client  78  cur=92.30  best=92.53\n",
      "Client  79  cur=99.00  best=99.00\n",
      "Client  80  cur=90.70  best=90.70\n",
      "Client  81  cur=90.87  best=90.87\n",
      "Client  82  cur=85.60  best= 0.00\n",
      "Client  83  cur=85.60  best= 0.00\n",
      "Client  84  cur=85.67  best= 0.00\n",
      "Client  85  cur=86.03  best=92.53\n",
      "Client  86  cur=85.67  best= 0.00\n",
      "Client  87  cur=95.73  best=95.83\n",
      "Client  88  cur=93.27  best=93.27\n",
      "Client  89  cur=85.67  best= 0.00\n",
      "Client  90  cur=85.60  best= 0.00\n",
      "Client  91  cur=52.00  best= 0.00\n",
      "Client  92  cur=98.13  best=98.13\n",
      "Client  93  cur=52.00  best= 0.00\n",
      "Client  94  cur=95.60  best=96.10\n",
      "Client  95  cur=85.60  best= 0.00\n",
      "Client  96  cur=87.07  best=95.83\n",
      "Client  97  cur=85.60  best= 0.00\n",
      "Client  98  cur=94.07  best=94.07\n",
      "Client  99  cur=97.47  best=97.73\n",
      "Round 4 AvgCur=85.87  AvgBest=57.53\n",
      "\n",
      "###### P2-ROUND 5 ######\n",
      "###### P2-ROUND 6 ######\n",
      "###### P2-ROUND 7 ######\n",
      "###### P2-ROUND 8 ######\n",
      "###### P2-ROUND 9 ######\n",
      "###### P2-ROUND 10 ######\n",
      "###### P2-ROUND 11 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0083\n",
      "Init Test Acc:  96.82 Final Test Acc: 95.15\n",
      "Client   0  cur=93.37  best=95.77\n",
      "Client   1  cur=52.00  best= 0.00\n",
      "Client   2  cur=94.33  best=98.17\n",
      "Client   3  cur=98.93  best=98.97\n",
      "Client   4  cur=95.67  best=95.73\n",
      "Client   5  cur=95.73  best=96.20\n",
      "Client   6  cur=82.23  best= 0.00\n",
      "Client   7  cur=95.57  best=95.77\n",
      "Client   8  cur=95.27  best=95.83\n",
      "Client   9  cur=92.03  best=92.03\n",
      "Client  10  cur=92.90  best=92.90\n",
      "Client  11  cur=97.37  best=98.17\n",
      "Client  12  cur=98.17  best=99.00\n",
      "Client  13  cur=92.73  best=96.20\n",
      "Client  14  cur=93.07  best=93.33\n",
      "Client  15  cur=92.93  best=96.00\n",
      "Client  16  cur=92.70  best=93.00\n",
      "Client  17  cur=95.27  best=95.83\n",
      "Client  18  cur=98.57  best=98.57\n",
      "Client  19  cur=99.07  best=99.07\n",
      "Client  20  cur=92.07  best=92.53\n",
      "Client  21  cur=98.10  best=98.30\n",
      "Client  22  cur=99.13  best=99.13\n",
      "Client  23  cur=94.30  best=94.30\n",
      "Client  24  cur=94.67  best=96.10\n",
      "Client  25  cur=95.83  best=95.83\n",
      "Client  26  cur=99.23  best=99.23\n",
      "Client  27  cur=62.27  best= 0.00\n",
      "Client  28  cur=98.17  best=98.17\n",
      "Client  29  cur=96.77  best=97.87\n",
      "Client  30  cur=95.87  best=96.00\n",
      "Client  31  cur=96.53  best=96.53\n",
      "Client  32  cur=97.97  best=98.30\n",
      "Client  33  cur=95.33  best=96.93\n",
      "Client  34  cur=99.00  best=99.17\n",
      "Client  35  cur=98.43  best=98.43\n",
      "Client  36  cur=98.37  best=98.47\n",
      "Client  37  cur=97.23  best=98.23\n",
      "Client  38  cur=95.73  best=95.97\n",
      "Client  39  cur=93.13  best=93.13\n",
      "Client  40  cur=99.03  best=99.17\n",
      "Client  41  cur=92.43  best=93.73\n",
      "Client  42  cur=86.13  best=92.53\n",
      "Client  43  cur=92.43  best=93.00\n",
      "Client  44  cur=97.80  best=97.80\n",
      "Client  45  cur=99.23  best=99.23\n",
      "Client  46  cur=92.07  best=95.77\n",
      "Client  47  cur=86.77  best=96.20\n",
      "Client  48  cur=92.80  best=92.80\n",
      "Client  49  cur=98.73  best=99.17\n",
      "Client  50  cur=95.83  best=95.90\n",
      "Client  51  cur=95.43  best=96.60\n",
      "Client  52  cur=95.67  best=96.60\n",
      "Client  53  cur=77.13  best=93.73\n",
      "Client  54  cur=93.17  best=95.83\n",
      "Client  55  cur=92.50  best=92.50\n",
      "Client  56  cur=98.07  best=98.07\n",
      "Client  57  cur=97.97  best=98.23\n",
      "Client  58  cur=95.70  best=95.70\n",
      "Client  59  cur=92.67  best=93.00\n",
      "Client  60  cur=95.13  best=96.93\n",
      "Client  61  cur=99.37  best=99.37\n",
      "Client  62  cur=98.17  best=98.17\n",
      "Client  63  cur=52.00  best= 0.00\n",
      "Client  64  cur=95.67  best=96.60\n",
      "Client  65  cur=94.97  best=94.97\n",
      "Client  66  cur=90.40  best=96.20\n",
      "Client  67  cur=97.00  best=98.17\n",
      "Client  68  cur=95.90  best=96.10\n",
      "Client  69  cur=79.03  best=79.03\n",
      "Client  70  cur=97.53  best=97.53\n",
      "Client  71  cur=91.27  best=93.73\n",
      "Client  72  cur=99.07  best=99.10\n",
      "Client  73  cur=99.13  best=99.13\n",
      "Client  74  cur=96.13  best=96.33\n",
      "Client  75  cur=62.27  best= 0.00\n",
      "Client  76  cur=96.00  best=96.00\n",
      "Client  77  cur=98.43  best=98.43\n",
      "Client  78  cur=92.30  best=92.53\n",
      "Client  79  cur=99.20  best=99.20\n",
      "Client  80  cur=91.33  best=91.33\n",
      "Client  81  cur=90.87  best=90.87\n",
      "Client  82  cur=93.10  best=93.10\n",
      "Client  83  cur=93.60  best=94.03\n",
      "Client  84  cur=95.83  best=96.93\n",
      "Client  85  cur=85.53  best=92.53\n",
      "Client  86  cur=96.07  best=96.93\n",
      "Client  87  cur=95.73  best=95.83\n",
      "Client  88  cur=93.40  best=93.73\n",
      "Client  89  cur=96.87  best=96.87\n",
      "Client  90  cur=85.60  best= 0.00\n",
      "Client  91  cur=98.90  best=98.90\n",
      "Client  92  cur=98.20  best=98.20\n",
      "Client  93  cur=98.93  best=99.17\n",
      "Client  94  cur=95.77  best=96.10\n",
      "Client  95  cur=93.33  best=93.33\n",
      "Client  96  cur=86.90  best=95.83\n",
      "Client  97  cur=95.93  best=95.97\n",
      "Client  98  cur=94.37  best=94.47\n",
      "Client  99  cur=97.53  best=97.73\n",
      "Round 11 AvgCur=93.24  AvgBest=90.28\n",
      "\n",
      "###### P2-ROUND 12 ######\n",
      "###### P2-ROUND 13 ######\n",
      "###### P2-ROUND 14 ######\n",
      "###### P2-ROUND 15 ######\n",
      "###### P2-ROUND 16 ######\n",
      "###### P2-ROUND 17 ######\n",
      "###### P2-ROUND 18 ######\n",
      "###### P2-ROUND 19 ######\n",
      "###### P2-ROUND 20 ######\n",
      "###### P2-ROUND 21 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0100\n",
      "Init Test Acc:  96.45 Final Test Acc: 95.18\n",
      "Client   0  cur=93.50  best=95.77\n",
      "Client   1  cur=98.90  best=98.90\n",
      "Client   2  cur=95.10  best=98.17\n",
      "Client   3  cur=99.10  best=99.10\n",
      "Client   4  cur=95.83  best=95.83\n",
      "Client   5  cur=95.73  best=96.20\n",
      "Client   6  cur=94.30  best=94.30\n",
      "Client   7  cur=95.53  best=95.97\n",
      "Client   8  cur=95.53  best=95.83\n",
      "Client   9  cur=92.10  best=92.10\n",
      "Client  10  cur=92.77  best=92.90\n",
      "Client  11  cur=97.53  best=98.17\n",
      "Client  12  cur=98.77  best=99.13\n",
      "Client  13  cur=93.20  best=96.20\n",
      "Client  14  cur=93.00  best=93.33\n",
      "Client  15  cur=93.87  best=96.27\n",
      "Client  16  cur=92.63  best=93.00\n",
      "Client  17  cur=95.87  best=95.87\n",
      "Client  18  cur=98.57  best=98.57\n",
      "Client  19  cur=99.13  best=99.13\n",
      "Client  20  cur=92.63  best=92.63\n",
      "Client  21  cur=98.27  best=98.30\n",
      "Client  22  cur=99.13  best=99.17\n",
      "Client  23  cur=94.87  best=94.90\n",
      "Client  24  cur=94.67  best=96.10\n",
      "Client  25  cur=95.77  best=95.87\n",
      "Client  26  cur=99.13  best=99.23\n",
      "Client  27  cur=90.57  best=90.57\n",
      "Client  28  cur=98.20  best=98.20\n",
      "Client  29  cur=96.77  best=97.87\n",
      "Client  30  cur=95.67  best=96.00\n",
      "Client  31  cur=96.37  best=96.53\n",
      "Client  32  cur=97.83  best=98.30\n",
      "Client  33  cur=95.60  best=96.93\n",
      "Client  34  cur=99.00  best=99.17\n",
      "Client  35  cur=98.40  best=98.63\n",
      "Client  36  cur=98.10  best=98.47\n",
      "Client  37  cur=97.07  best=98.23\n",
      "Client  38  cur=96.00  best=96.00\n",
      "Client  39  cur=92.73  best=93.13\n",
      "Client  40  cur=99.10  best=99.17\n",
      "Client  41  cur=92.13  best=93.73\n",
      "Client  42  cur=86.40  best=92.53\n",
      "Client  43  cur=93.00  best=93.43\n",
      "Client  44  cur=98.17  best=98.23\n",
      "Client  45  cur=99.27  best=99.27\n",
      "Client  46  cur=91.37  best=95.97\n",
      "Client  47  cur=86.77  best=96.20\n",
      "Client  48  cur=92.07  best=92.80\n",
      "Client  49  cur=99.03  best=99.17\n",
      "Client  50  cur=95.97  best=96.27\n",
      "Client  51  cur=95.53  best=96.60\n",
      "Client  52  cur=95.50  best=96.60\n",
      "Client  53  cur=78.90  best=93.73\n",
      "Client  54  cur=94.70  best=95.83\n",
      "Client  55  cur=92.77  best=92.77\n",
      "Client  56  cur=98.13  best=98.13\n",
      "Client  57  cur=98.07  best=98.23\n",
      "Client  58  cur=95.53  best=95.70\n",
      "Client  59  cur=92.67  best=93.00\n",
      "Client  60  cur=95.17  best=96.93\n",
      "Client  61  cur=99.33  best=99.40\n",
      "Client  62  cur=98.37  best=98.37\n",
      "Client  63  cur=99.40  best=99.40\n",
      "Client  64  cur=96.17  best=96.60\n",
      "Client  65  cur=95.37  best=96.27\n",
      "Client  66  cur=90.40  best=96.20\n",
      "Client  67  cur=96.97  best=98.17\n",
      "Client  68  cur=95.70  best=96.10\n",
      "Client  69  cur=80.17  best=80.17\n",
      "Client  70  cur=97.80  best=97.80\n",
      "Client  71  cur=91.13  best=93.73\n",
      "Client  72  cur=99.07  best=99.10\n",
      "Client  73  cur=99.20  best=99.20\n",
      "Client  74  cur=96.60  best=96.60\n",
      "Client  75  cur=93.10  best=93.30\n",
      "Client  76  cur=95.97  best=96.27\n",
      "Client  77  cur=98.20  best=98.43\n",
      "Client  78  cur=92.03  best=92.53\n",
      "Client  79  cur=99.20  best=99.20\n",
      "Client  80  cur=91.83  best=93.30\n",
      "Client  81  cur=91.33  best=91.33\n",
      "Client  82  cur=93.57  best=93.57\n",
      "Client  83  cur=93.63  best=95.97\n",
      "Client  84  cur=96.17  best=96.93\n",
      "Client  85  cur=85.43  best=93.30\n",
      "Client  86  cur=96.43  best=96.93\n",
      "Client  87  cur=95.60  best=95.83\n",
      "Client  88  cur=93.37  best=93.73\n",
      "Client  89  cur=96.90  best=96.90\n",
      "Client  90  cur=95.20  best=95.23\n",
      "Client  91  cur=98.97  best=99.03\n",
      "Client  92  cur=98.27  best=98.27\n",
      "Client  93  cur=98.93  best=99.17\n",
      "Client  94  cur=95.77  best=96.10\n",
      "Client  95  cur=94.10  best=95.97\n",
      "Client  96  cur=85.17  best=95.83\n",
      "Client  97  cur=95.93  best=95.97\n",
      "Client  98  cur=94.60  best=94.60\n",
      "Client  99  cur=97.53  best=97.73\n",
      "Round 21 AvgCur=95.11  AvgBest=96.16\n",
      "\n",
      "###### P2-ROUND 22 ######\n",
      "###### P2-ROUND 23 ######\n",
      "###### P2-ROUND 24 ######\n",
      "###### P2-ROUND 25 ######\n",
      "###### P2-ROUND 26 ######\n",
      "###### P2-ROUND 27 ######\n",
      "###### P2-ROUND 28 ######\n",
      "###### P2-ROUND 29 ######\n",
      "###### P2-ROUND 30 ######\n",
      "###### P2-ROUND 31 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0082\n",
      "Init Test Acc:  94.91 Final Test Acc: 94.81\n",
      "Client   0  cur=93.83  best=96.00\n",
      "Client   1  cur=98.77  best=98.90\n",
      "Client   2  cur=94.63  best=98.17\n",
      "Client   3  cur=99.10  best=99.10\n",
      "Client   4  cur=95.77  best=96.03\n",
      "Client   5  cur=95.53  best=96.20\n",
      "Client   6  cur=95.07  best=95.93\n",
      "Client   7  cur=95.90  best=96.00\n",
      "Client   8  cur=95.67  best=95.83\n",
      "Client   9  cur=92.00  best=92.10\n",
      "Client  10  cur=92.87  best=92.90\n",
      "Client  11  cur=97.53  best=98.17\n",
      "Client  12  cur=98.97  best=99.13\n",
      "Client  13  cur=93.23  best=96.20\n",
      "Client  14  cur=92.97  best=93.33\n",
      "Client  15  cur=94.03  best=96.27\n",
      "Client  16  cur=91.37  best=93.00\n",
      "Client  17  cur=96.00  best=96.00\n",
      "Client  18  cur=98.43  best=98.57\n",
      "Client  19  cur=99.13  best=99.17\n",
      "Client  20  cur=92.60  best=92.63\n",
      "Client  21  cur=98.37  best=98.40\n",
      "Client  22  cur=99.13  best=99.17\n",
      "Client  23  cur=94.33  best=94.90\n",
      "Client  24  cur=94.67  best=96.10\n",
      "Client  25  cur=96.03  best=96.03\n",
      "Client  26  cur=99.13  best=99.23\n",
      "Client  27  cur=89.93  best=90.57\n",
      "Client  28  cur=98.50  best=98.50\n",
      "Client  29  cur=96.77  best=97.87\n",
      "Client  30  cur=96.00  best=97.07\n",
      "Client  31  cur=96.73  best=97.07\n",
      "Client  32  cur=98.10  best=98.30\n",
      "Client  33  cur=95.80  best=96.93\n",
      "Client  34  cur=98.93  best=99.17\n",
      "Client  35  cur=98.40  best=98.63\n",
      "Client  36  cur=98.33  best=98.60\n",
      "Client  37  cur=97.27  best=98.23\n",
      "Client  38  cur=96.03  best=96.03\n",
      "Client  39  cur=92.63  best=93.13\n",
      "Client  40  cur=99.10  best=99.17\n",
      "Client  41  cur=92.07  best=93.73\n",
      "Client  42  cur=85.93  best=92.53\n",
      "Client  43  cur=92.90  best=93.43\n",
      "Client  44  cur=98.17  best=98.30\n",
      "Client  45  cur=99.27  best=99.27\n",
      "Client  46  cur=92.63  best=96.00\n",
      "Client  47  cur=88.30  best=96.20\n",
      "Client  48  cur=92.00  best=92.80\n",
      "Client  49  cur=98.93  best=99.17\n",
      "Client  50  cur=96.33  best=97.07\n",
      "Client  51  cur=95.57  best=97.07\n",
      "Client  52  cur=95.93  best=96.60\n",
      "Client  53  cur=77.27  best=93.73\n",
      "Client  54  cur=94.07  best=95.83\n",
      "Client  55  cur=92.53  best=92.77\n",
      "Client  56  cur=98.13  best=98.13\n",
      "Client  57  cur=98.07  best=98.23\n",
      "Client  58  cur=95.53  best=95.70\n",
      "Client  59  cur=92.40  best=93.00\n",
      "Client  60  cur=94.80  best=96.93\n",
      "Client  61  cur=99.27  best=99.40\n",
      "Client  62  cur=98.23  best=98.37\n",
      "Client  63  cur=99.33  best=99.40\n",
      "Client  64  cur=96.03  best=96.60\n",
      "Client  65  cur=95.37  best=96.27\n",
      "Client  66  cur=85.90  best=96.20\n",
      "Client  67  cur=96.90  best=98.17\n",
      "Client  68  cur=96.00  best=96.10\n",
      "Client  69  cur=78.60  best=90.07\n",
      "Client  70  cur=98.07  best=98.10\n",
      "Client  71  cur=91.20  best=93.73\n",
      "Client  72  cur=99.00  best=99.10\n",
      "Client  73  cur=99.17  best=99.20\n",
      "Client  74  cur=96.30  best=96.60\n",
      "Client  75  cur=93.47  best=93.47\n",
      "Client  76  cur=96.07  best=96.27\n",
      "Client  77  cur=98.20  best=98.43\n",
      "Client  78  cur=92.77  best=92.77\n",
      "Client  79  cur=99.13  best=99.20\n",
      "Client  80  cur=91.90  best=93.30\n",
      "Client  81  cur=90.90  best=91.33\n",
      "Client  82  cur=93.57  best=93.57\n",
      "Client  83  cur=93.97  best=95.97\n",
      "Client  84  cur=95.80  best=96.93\n",
      "Client  85  cur=85.87  best=93.30\n",
      "Client  86  cur=96.60  best=96.93\n",
      "Client  87  cur=95.60  best=95.83\n",
      "Client  88  cur=93.70  best=93.73\n",
      "Client  89  cur=96.67  best=96.90\n",
      "Client  90  cur=95.60  best=95.60\n",
      "Client  91  cur=99.20  best=99.20\n",
      "Client  92  cur=98.37  best=98.37\n",
      "Client  93  cur=98.90  best=99.17\n",
      "Client  94  cur=95.83  best=96.10\n",
      "Client  95  cur=94.33  best=96.00\n",
      "Client  96  cur=87.73  best=95.83\n",
      "Client  97  cur=94.43  best=95.97\n",
      "Client  98  cur=94.27  best=94.60\n",
      "Client  99  cur=97.67  best=97.73\n",
      "Round 31 AvgCur=95.08  AvgBest=96.33\n",
      "\n",
      "###### P2-ROUND 32 ######\n",
      "###### P2-ROUND 33 ######\n",
      "###### P2-ROUND 34 ######\n",
      "###### P2-ROUND 35 ######\n",
      "###### P2-ROUND 36 ######\n",
      "###### P2-ROUND 37 ######\n",
      "###### P2-ROUND 38 ######\n",
      "###### P2-ROUND 39 ######\n",
      "###### P2-ROUND 40 ######\n",
      "###### P2-ROUND 41 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0055\n",
      "Init Test Acc:  93.81 Final Test Acc: 95.38\n",
      "Client   0  cur=93.67  best=96.07\n",
      "Client   1  cur=98.97  best=99.37\n",
      "Client   2  cur=94.90  best=98.17\n",
      "Client   3  cur=99.30  best=99.30\n",
      "Client   4  cur=95.77  best=96.10\n",
      "Client   5  cur=95.70  best=96.20\n",
      "Client   6  cur=95.33  best=96.40\n",
      "Client   7  cur=95.73  best=96.00\n",
      "Client   8  cur=95.97  best=96.07\n",
      "Client   9  cur=92.23  best=92.63\n",
      "Client  10  cur=92.97  best=92.97\n",
      "Client  11  cur=97.53  best=98.17\n",
      "Client  12  cur=98.97  best=99.13\n",
      "Client  13  cur=92.93  best=96.20\n",
      "Client  14  cur=92.77  best=93.33\n",
      "Client  15  cur=95.13  best=96.27\n",
      "Client  16  cur=91.37  best=93.00\n",
      "Client  17  cur=95.13  best=96.00\n",
      "Client  18  cur=98.57  best=98.93\n",
      "Client  19  cur=99.13  best=99.17\n",
      "Client  20  cur=92.53  best=92.63\n",
      "Client  21  cur=98.10  best=98.40\n",
      "Client  22  cur=99.13  best=99.17\n",
      "Client  23  cur=94.30  best=94.90\n",
      "Client  24  cur=95.10  best=96.10\n",
      "Client  25  cur=96.10  best=96.10\n",
      "Client  26  cur=99.17  best=99.23\n",
      "Client  27  cur=89.93  best=90.57\n",
      "Client  28  cur=98.50  best=98.50\n",
      "Client  29  cur=96.93  best=97.87\n",
      "Client  30  cur=96.23  best=97.07\n",
      "Client  31  cur=96.73  best=97.07\n",
      "Client  32  cur=98.40  best=98.50\n",
      "Client  33  cur=95.67  best=96.93\n",
      "Client  34  cur=98.87  best=99.37\n",
      "Client  35  cur=98.43  best=98.63\n",
      "Client  36  cur=98.23  best=98.60\n",
      "Client  37  cur=97.43  best=98.23\n",
      "Client  38  cur=95.50  best=96.03\n",
      "Client  39  cur=92.63  best=93.13\n",
      "Client  40  cur=99.13  best=99.37\n",
      "Client  41  cur=92.40  best=93.73\n",
      "Client  42  cur=86.63  best=92.53\n",
      "Client  43  cur=93.03  best=93.43\n",
      "Client  44  cur=98.13  best=98.30\n",
      "Client  45  cur=99.30  best=99.37\n",
      "Client  46  cur=91.93  best=96.00\n",
      "Client  47  cur=88.87  best=96.20\n",
      "Client  48  cur=92.13  best=92.80\n",
      "Client  49  cur=99.07  best=99.17\n",
      "Client  50  cur=96.33  best=97.07\n",
      "Client  51  cur=95.40  best=97.07\n",
      "Client  52  cur=96.20  best=96.60\n",
      "Client  53  cur=76.13  best=93.73\n",
      "Client  54  cur=94.07  best=95.83\n",
      "Client  55  cur=92.97  best=92.97\n",
      "Client  56  cur=98.13  best=98.23\n",
      "Client  57  cur=98.13  best=98.23\n",
      "Client  58  cur=94.70  best=95.70\n",
      "Client  59  cur=93.00  best=93.00\n",
      "Client  60  cur=95.13  best=96.97\n",
      "Client  61  cur=99.33  best=99.40\n",
      "Client  62  cur=98.23  best=98.37\n",
      "Client  63  cur=99.37  best=99.40\n",
      "Client  64  cur=95.77  best=96.97\n",
      "Client  65  cur=93.90  best=96.27\n",
      "Client  66  cur=85.90  best=96.20\n",
      "Client  67  cur=97.13  best=98.17\n",
      "Client  68  cur=95.67  best=96.10\n",
      "Client  69  cur=80.20  best=90.07\n",
      "Client  70  cur=98.03  best=98.10\n",
      "Client  71  cur=91.67  best=93.73\n",
      "Client  72  cur=99.10  best=99.13\n",
      "Client  73  cur=99.20  best=99.20\n",
      "Client  74  cur=96.47  best=96.60\n",
      "Client  75  cur=93.13  best=93.47\n",
      "Client  76  cur=96.43  best=96.97\n",
      "Client  77  cur=98.43  best=98.43\n",
      "Client  78  cur=92.40  best=92.77\n",
      "Client  79  cur=99.20  best=99.23\n",
      "Client  80  cur=91.77  best=93.30\n",
      "Client  81  cur=90.30  best=91.57\n",
      "Client  82  cur=94.20  best=94.20\n",
      "Client  83  cur=93.63  best=95.97\n",
      "Client  84  cur=96.23  best=96.93\n",
      "Client  85  cur=85.00  best=93.30\n",
      "Client  86  cur=96.60  best=96.93\n",
      "Client  87  cur=95.60  best=95.83\n",
      "Client  88  cur=93.70  best=93.73\n",
      "Client  89  cur=96.70  best=96.97\n",
      "Client  90  cur=95.13  best=95.70\n",
      "Client  91  cur=99.20  best=99.30\n",
      "Client  92  cur=98.37  best=98.47\n",
      "Client  93  cur=98.90  best=99.17\n",
      "Client  94  cur=95.63  best=96.10\n",
      "Client  95  cur=94.20  best=96.00\n",
      "Client  96  cur=86.57  best=95.83\n",
      "Client  97  cur=95.47  best=95.97\n",
      "Client  98  cur=94.97  best=94.97\n",
      "Client  99  cur=97.27  best=97.73\n",
      "Round 41 AvgCur=95.10  AvgBest=96.39\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# ---------------------------------------------------------------------\n",
    "#  PRE-REQS (already defined earlier in your code base)\n",
    "#  --------------------------------------------------------------------\n",
    "# clusters               : List[List[int]]   – client IDs per cluster\n",
    "# clients                : List[Client_ClusterFL]\n",
    "# clients[k].net         : CombinedModelMNIST (4-component model)\n",
    "# clients_clust_id       : Dict[client_id → cluster_id]\n",
    "# net_dataidx_map        : Dict[client_id → #samples]  (for weighting)\n",
    "# FedAvg2(state_dict_list, weight_list)      – utility\n",
    "# ---------------------------------------------------------------------\n",
    "\n",
    "import numpy as np, copy, gc\n",
    "\n",
    "# ---------------------------------------------------------------------\n",
    "# 1.  Initialise per-cluster cache (trainable parts only)\n",
    "# ---------------------------------------------------------------------\n",
    "cluster_global_trainable = {\n",
    "    c: {\n",
    "        'encoder':    copy.deepcopy(\n",
    "            clients[clusters[c][0]].net.own_encoder.state_dict()),\n",
    "        'classifier': copy.deepcopy(\n",
    "            clients[clusters[c][0]].net.own_classifier.state_dict()),\n",
    "    }\n",
    "    for c in range(len(clusters))\n",
    "}\n",
    "\n",
    "# ---------------------------------------------------------------------\n",
    "# 2.  Phase-2 FedAvg loop  (encoder + classifier)\n",
    "# ---------------------------------------------------------------------\n",
    "flag = 10                    # print interval after warm-up\n",
    "loss_locals2 = []\n",
    "init_local_tacc2, init_local_tloss2  = [], []\n",
    "final_local_tacc2, final_local_tloss2 = [], []\n",
    "clients2_best_acc2 = [0 for _ in range(args.num_users)]\n",
    "\n",
    "for round_idx in range(41):\n",
    "\n",
    "    print(f'###### P2-ROUND {round_idx+1} ######')\n",
    "\n",
    "    # ----- 2.1  sample participants ----------------------------------\n",
    "    idxs_users = np.random.choice(\n",
    "        range(args.num_users),\n",
    "        int(args.frac * args.num_users),\n",
    "        replace=False\n",
    "    )\n",
    "\n",
    "    # ----- 2.2  broadcast current cluster weights -------------------\n",
    "    for uid in idxs_users:\n",
    "        cid = clients_clust_id[uid]\n",
    "        clients[uid].net.own_encoder.load_state_dict(\n",
    "            cluster_global_trainable[cid]['encoder'])\n",
    "        clients[uid].net.own_classifier.load_state_dict(\n",
    "            cluster_global_trainable[cid]['classifier'])\n",
    "\n",
    "    # ----- 2.3  book-keeping ----------------------------------------\n",
    "    idx_clusters_round = {c: [] for c in range(len(clusters))}\n",
    "    for uid in idxs_users:\n",
    "        idx_clusters_round[clients_clust_id[uid]].append(uid)\n",
    "\n",
    "    # containers for FedAvg\n",
    "    cluster_encoders, cluster_classifiers, cluster_freqs = {}, {}, {}\n",
    "\n",
    "    # ----- 2.4  local training --------------------------------------\n",
    "    for uid in idxs_users:\n",
    "        cid = clients_clust_id[uid]\n",
    "\n",
    "        # metrics before local train\n",
    "        if round_idx % flag == 0:\n",
    "            l0, a0 = clients[uid].eval_test()\n",
    "            init_local_tloss2.append(l0)\n",
    "            init_local_tacc2.append(a0)\n",
    "            clients2_best_acc2[uid] = max(clients2_best_acc2[uid], a0)\n",
    "\n",
    "        # local update (trainable own-encoder + own-classifier)\n",
    "        loss = clients[uid].train2()\n",
    "        loss_locals2.append(loss)\n",
    "\n",
    "        # metrics after local train\n",
    "        l1, a1 = clients[uid].eval_test()\n",
    "        final_local_tloss2.append(l1)\n",
    "        final_local_tacc2.append(a1)\n",
    "        clients2_best_acc2[uid] = max(clients2_best_acc2[uid], a1)\n",
    "\n",
    "        # collect trainable parts for FedAvg\n",
    "        cluster_encoders   .setdefault(cid, []).append(\n",
    "            copy.deepcopy(clients[uid].net.own_encoder.state_dict()))\n",
    "        cluster_classifiers.setdefault(cid, []).append(\n",
    "            copy.deepcopy(clients[uid].net.own_classifier.state_dict()))\n",
    "        cluster_freqs      .setdefault(cid, []).append(\n",
    "            len(net_dataidx_map[uid]))\n",
    "\n",
    "    # ----- 2.5  FedAvg encoder & classifier per cluster --------------\n",
    "    for cid in cluster_encoders:          # same keys as classifiers\n",
    "        total_samples = sum(cluster_freqs[cid])\n",
    "        weights = [n / total_samples for n in cluster_freqs[cid]]\n",
    "\n",
    "        cluster_global_trainable[cid]['encoder'] = \\\n",
    "            FedAvg2(cluster_encoders[cid], weights)\n",
    "\n",
    "        cluster_global_trainable[cid]['classifier'] = \\\n",
    "            FedAvg2(cluster_classifiers[cid], weights)\n",
    "\n",
    "    # ----- 2.6  pretty printing every `flag` rounds ------------------\n",
    "    if round_idx < 4: flag = 1\n",
    "    else:             flag = 10\n",
    "\n",
    "    if round_idx % flag == 0:\n",
    "        print('## END-OF-ROUND-STATS ##')\n",
    "        print(f'Avg train loss: {np.mean(loss_locals2):.4f}')\n",
    "        print(f'Init Test Acc:  {np.mean(init_local_tacc2):.2f}',\n",
    "              f'Final Test Acc: {np.mean(final_local_tacc2):.2f}')\n",
    "\n",
    "        # per-client summary\n",
    "        cur_acc = []\n",
    "        for uid in range(args.num_users):\n",
    "            _, acc = clients[uid].eval_test()\n",
    "            cur_acc.append(acc)\n",
    "            print(f'Client {uid:3d}  cur={acc:5.2f}  best={clients2_best_acc2[uid]:5.2f}')\n",
    "\n",
    "        print(f'Round {round_idx+1} AvgCur={np.mean(cur_acc):.2f}  '\n",
    "              f'AvgBest={np.mean(clients2_best_acc2):.2f}\\n')\n",
    "\n",
    "        # clear trackers for next flagged round\n",
    "        loss_locals2.clear()\n",
    "        init_local_tacc2.clear(); init_local_tloss2.clear()\n",
    "        final_local_tacc2.clear(); final_local_tloss2.clear()\n",
    "        gc.collect()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "###### P2-ROUND 1 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0104\n",
      "Init Test Acc:  93.79 Final Test Acc: 92.73\n",
      "Client   0  cur=98.63  best=98.63\n",
      "Client   1  cur=98.83  best=99.03\n",
      "Client   2  cur=99.27  best=99.27\n",
      "Client   3  cur=92.87  best=96.67\n",
      "Client   4  cur=84.90  best=90.63\n",
      "Client   5  cur=98.93  best=99.17\n",
      "Client   6  cur=98.70  best=99.07\n",
      "Client   7  cur=90.90  best=93.43\n",
      "Client   8  cur=90.80  best=90.80\n",
      "Client   9  cur=99.03  best=99.03\n",
      "Client  10  cur=97.53  best=98.90\n",
      "Client  11  cur=99.00  best=99.00\n",
      "Client  12  cur=96.23  best=96.70\n",
      "Client  13  cur=98.67  best=98.87\n",
      "Client  14  cur=99.07  best=99.07\n",
      "Client  15  cur=95.77  best=96.67\n",
      "Client  16  cur=89.80  best=90.37\n",
      "Client  17  cur=97.17  best=99.37\n",
      "Client  18  cur=96.50  best=99.37\n",
      "Client  19  cur=98.97  best=98.97\n",
      "Client  20  cur=87.20  best=90.63\n",
      "Client  21  cur=96.27  best=96.27\n",
      "Client  22  cur=91.97  best=92.63\n",
      "Client  23  cur=96.97  best=96.97\n",
      "Client  24  cur=88.83  best=90.87\n",
      "Client  25  cur=86.30  best=90.73\n",
      "Client  26  cur=92.23  best=92.87\n",
      "Client  27  cur=96.23  best=99.37\n",
      "Client  28  cur=90.07  best=92.87\n",
      "Client  29  cur=96.57  best=96.93\n",
      "Client  30  cur=88.03  best=90.87\n",
      "Client  31  cur=96.10  best=96.67\n",
      "Client  32  cur=99.23  best=99.37\n",
      "Client  33  cur=90.97  best=90.97\n",
      "Client  34  cur=96.20  best=96.87\n",
      "Client  35  cur=96.37  best=96.73\n",
      "Client  36  cur=95.57  best=96.47\n",
      "Client  37  cur=95.20  best=96.87\n",
      "Client  38  cur=98.97  best=99.00\n",
      "Client  39  cur=88.60  best=90.03\n",
      "Client  40  cur=98.67  best=99.07\n",
      "Client  41  cur=91.17  best=93.00\n",
      "Client  42  cur=90.40  best=90.87\n",
      "Client  43  cur=89.97  best=92.87\n",
      "Client  44  cur=88.37  best=92.40\n",
      "Client  45  cur=98.87  best=99.07\n",
      "Client  46  cur=90.53  best=90.73\n",
      "Client  47  cur=99.20  best=99.37\n",
      "Client  48  cur=98.90  best=98.97\n",
      "Client  49  cur=87.83  best=88.67\n",
      "Client  50  cur=97.83  best=99.30\n",
      "Client  51  cur=89.37  best=93.00\n",
      "Client  52  cur=94.67  best=96.77\n",
      "Client  53  cur=85.67  best=90.87\n",
      "Client  54  cur=91.97  best=93.43\n",
      "Client  55  cur=89.80  best=90.87\n",
      "Client  56  cur=94.63  best=96.77\n",
      "Client  57  cur=86.87  best=92.87\n",
      "Client  58  cur=99.17  best=99.37\n",
      "Client  59  cur=89.53  best=90.87\n",
      "Client  60  cur=99.10  best=99.10\n",
      "Client  61  cur=96.63  best=96.93\n",
      "Client  62  cur=95.90  best=96.67\n",
      "Client  63  cur=99.27  best=99.27\n",
      "Client  64  cur=99.27  best=99.30\n",
      "Client  65  cur=92.07  best=92.63\n",
      "Client  66  cur=84.23  best=90.77\n",
      "Client  67  cur=98.83  best=99.00\n",
      "Client  68  cur=85.23  best=90.87\n",
      "Client  69  cur=95.83  best=96.93\n",
      "Client  70  cur=98.80  best=98.97\n",
      "Client  71  cur=92.40  best=92.87\n",
      "Client  72  cur=99.27  best=99.37\n",
      "Client  73  cur=90.33  best=90.87\n",
      "Client  74  cur=88.47  best=92.87\n",
      "Client  75  cur=96.40  best=96.67\n",
      "Client  76  cur=91.80  best=93.00\n",
      "Client  77  cur=83.63  best=90.73\n",
      "Client  78  cur=96.80  best=96.97\n",
      "Client  79  cur=96.57  best=96.87\n",
      "Client  80  cur=98.87  best=99.07\n",
      "Client  81  cur=96.87  best=96.87\n",
      "Client  82  cur=90.33  best=91.40\n",
      "Client  83  cur=98.37  best=99.23\n",
      "Client  84  cur=89.53  best=90.63\n",
      "Client  85  cur=89.73  best=90.03\n",
      "Client  86  cur=98.97  best=99.00\n",
      "Client  87  cur=98.90  best=99.37\n",
      "Client  88  cur=99.03  best=99.03\n",
      "Client  89  cur=98.87  best=99.30\n",
      "Client  90  cur=92.60  best=93.43\n",
      "Client  91  cur=95.50  best=96.93\n",
      "Client  92  cur=99.20  best=99.30\n",
      "Client  93  cur=90.30  best=91.43\n",
      "Client  94  cur=94.37  best=94.53\n",
      "Client  95  cur=96.93  best=98.00\n",
      "Client  96  cur=95.63  best=96.70\n",
      "Client  97  cur=99.30  best=99.33\n",
      "Client  98  cur=99.17  best=99.30\n",
      "Client  99  cur=99.23  best=99.33\n",
      "Round 1 AvgCur=94.45  AvgBest=95.69\n",
      "\n",
      "###### P2-ROUND 2 ######\n",
      "## END-OF-ROUND-STATS ##\n",
      "Avg train loss: 0.0079\n",
      "Init Test Acc:  95.98 Final Test Acc: 94.93\n",
      "Client   0  cur=98.63  best=98.63\n",
      "Client   1  cur=98.83  best=99.03\n",
      "Client   2  cur=99.27  best=99.27\n",
      "Client   3  cur=91.83  best=96.80\n",
      "Client   4  cur=84.90  best=90.63\n",
      "Client   5  cur=98.93  best=99.17\n",
      "Client   6  cur=98.70  best=99.07\n",
      "Client   7  cur=90.90  best=93.43\n",
      "Client   8  cur=90.80  best=90.80\n",
      "Client   9  cur=99.03  best=99.03\n",
      "Client  10  cur=97.53  best=98.90\n",
      "Client  11  cur=99.00  best=99.00\n",
      "Client  12  cur=96.23  best=96.70\n",
      "Client  13  cur=98.67  best=98.87\n",
      "Client  14  cur=99.07  best=99.07\n",
      "Client  15  cur=95.77  best=96.67\n",
      "Client  16  cur=89.80  best=90.37\n",
      "Client  17  cur=97.17  best=99.37\n",
      "Client  18  cur=96.50  best=99.37\n",
      "Client  19  cur=98.97  best=98.97\n",
      "Client  20  cur=87.20  best=90.63\n",
      "Client  21  cur=96.27  best=96.27\n",
      "Client  22  cur=91.97  best=92.63\n",
      "Client  23  cur=96.97  best=96.97\n",
      "Client  24  cur=88.83  best=90.87\n",
      "Client  25  cur=86.30  best=90.73\n",
      "Client  26  cur=92.23  best=92.87\n",
      "Client  27  cur=96.23  best=99.37\n",
      "Client  28  cur=87.90  best=92.87\n",
      "Client  29  cur=96.57  best=96.93\n",
      "Client  30  cur=88.10  best=90.87\n",
      "Client  31  cur=96.10  best=96.67\n",
      "Client  32  cur=99.23  best=99.37\n",
      "Client  33  cur=90.97  best=90.97\n",
      "Client  34  cur=96.33  best=96.87\n",
      "Client  35  cur=95.90  best=96.80\n",
      "Client  36  cur=95.70  best=96.80\n",
      "Client  37  cur=95.17  best=96.87\n",
      "Client  38  cur=98.97  best=99.00\n",
      "Client  39  cur=88.60  best=90.03\n",
      "Client  40  cur=98.67  best=99.07\n",
      "Client  41  cur=91.17  best=93.00\n",
      "Client  42  cur=90.40  best=90.87\n",
      "Client  43  cur=89.97  best=92.87\n",
      "Client  44  cur=88.37  best=92.40\n",
      "Client  45  cur=98.87  best=99.07\n",
      "Client  46  cur=90.53  best=90.73\n",
      "Client  47  cur=99.20  best=99.37\n",
      "Client  48  cur=98.90  best=98.97\n",
      "Client  49  cur=87.83  best=88.67\n",
      "Client  50  cur=97.83  best=99.30\n",
      "Client  51  cur=89.37  best=93.00\n",
      "Client  52  cur=94.67  best=96.77\n",
      "Client  53  cur=85.67  best=90.87\n",
      "Client  54  cur=91.97  best=93.43\n",
      "Client  55  cur=89.80  best=90.87\n",
      "Client  56  cur=94.63  best=96.77\n",
      "Client  57  cur=86.87  best=92.87\n",
      "Client  58  cur=99.20  best=99.37\n",
      "Client  59  cur=89.83  best=90.87\n",
      "Client  60  cur=99.10  best=99.10\n",
      "Client  61  cur=96.63  best=96.93\n",
      "Client  62  cur=95.90  best=96.67\n",
      "Client  63  cur=99.10  best=99.27\n",
      "Client  64  cur=99.27  best=99.30\n",
      "Client  65  cur=93.07  best=93.07\n",
      "Client  66  cur=84.23  best=90.77\n",
      "Client  67  cur=98.83  best=99.00\n",
      "Client  68  cur=85.23  best=90.87\n",
      "Client  69  cur=95.83  best=96.93\n",
      "Client  70  cur=98.80  best=98.97\n",
      "Client  71  cur=92.40  best=92.87\n",
      "Client  72  cur=99.27  best=99.37\n",
      "Client  73  cur=90.33  best=90.87\n",
      "Client  74  cur=88.47  best=92.87\n",
      "Client  75  cur=96.40  best=96.67\n",
      "Client  76  cur=92.00  best=93.00\n",
      "Client  77  cur=83.63  best=90.73\n",
      "Client  78  cur=97.03  best=97.03\n",
      "Client  79  cur=96.57  best=96.87\n",
      "Client  80  cur=98.87  best=99.07\n",
      "Client  81  cur=96.77  best=96.87\n",
      "Client  82  cur=90.33  best=91.40\n",
      "Client  83  cur=98.37  best=99.23\n",
      "Client  84  cur=89.53  best=90.63\n",
      "Client  85  cur=89.73  best=90.03\n",
      "Client  86  cur=98.97  best=99.00\n",
      "Client  87  cur=98.90  best=99.37\n",
      "Client  88  cur=99.03  best=99.03\n",
      "Client  89  cur=98.87  best=99.30\n",
      "Client  90  cur=92.60  best=93.43\n",
      "Client  91  cur=95.50  best=96.93\n",
      "Client  92  cur=99.17  best=99.30\n",
      "Client  93  cur=90.30  best=91.43\n",
      "Client  94  cur=94.37  best=94.53\n",
      "Client  95  cur=97.27  best=98.90\n",
      "Client  96  cur=96.03  best=96.80\n",
      "Client  97  cur=99.30  best=99.33\n",
      "Client  98  cur=99.17  best=99.30\n",
      "Client  99  cur=99.23  best=99.33\n",
      "Round 2 AvgCur=94.44  AvgBest=95.71\n",
      "\n",
      "###### P2-ROUND 3 ######\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[28], line 40\u001b[0m\n\u001b[1;32m     37\u001b[0m     clients2_best_acc2[uid] \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mmax\u001b[39m(clients2_best_acc2[uid], a0)\n\u001b[1;32m     39\u001b[0m \u001b[38;5;66;03m# local update (trainable own-encoder + own-classifier)\u001b[39;00m\n\u001b[0;32m---> 40\u001b[0m loss \u001b[38;5;241m=\u001b[39m \u001b[43mclients\u001b[49m\u001b[43m[\u001b[49m\u001b[43muid\u001b[49m\u001b[43m]\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mtrain2\u001b[49m\u001b[43m(\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     41\u001b[0m loss_locals2\u001b[38;5;241m.\u001b[39mappend(loss)\n\u001b[1;32m     43\u001b[0m \u001b[38;5;66;03m# metrics after local train\u001b[39;00m\n",
      "File \u001b[0;32m~/Misc/PACFLComboNB/Flag/src/client/client_cluster_fl.py:91\u001b[0m, in \u001b[0;36mClient_ClusterFL.train2\u001b[0;34m(self, is_print)\u001b[0m\n\u001b[1;32m     89\u001b[0m optimizer\u001b[38;5;241m.\u001b[39mzero_grad()\n\u001b[1;32m     90\u001b[0m logits \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mnet(images)\n\u001b[0;32m---> 91\u001b[0m loss   \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mloss_func\u001b[49m\u001b[43m(\u001b[49m\u001b[43mlogits\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mlabels\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     92\u001b[0m loss\u001b[38;5;241m.\u001b[39mbackward()\n\u001b[1;32m     93\u001b[0m optimizer\u001b[38;5;241m.\u001b[39mstep()\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/module.py:1532\u001b[0m, in \u001b[0;36mModule._wrapped_call_impl\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1530\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_compiled_call_impl(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)  \u001b[38;5;66;03m# type: ignore[misc]\u001b[39;00m\n\u001b[1;32m   1531\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[0;32m-> 1532\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_call_impl\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/module.py:1541\u001b[0m, in \u001b[0;36mModule._call_impl\u001b[0;34m(self, *args, **kwargs)\u001b[0m\n\u001b[1;32m   1536\u001b[0m \u001b[38;5;66;03m# If we don't have any hooks, we want to skip the rest of the logic in\u001b[39;00m\n\u001b[1;32m   1537\u001b[0m \u001b[38;5;66;03m# this function, and just call forward.\u001b[39;00m\n\u001b[1;32m   1538\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m (\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_pre_hooks\n\u001b[1;32m   1539\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_backward_hooks\n\u001b[1;32m   1540\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_forward_pre_hooks):\n\u001b[0;32m-> 1541\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mforward_call\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43margs\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[38;5;241;43m*\u001b[39;49m\u001b[43mkwargs\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1543\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[1;32m   1544\u001b[0m     result \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/modules/loss.py:1185\u001b[0m, in \u001b[0;36mCrossEntropyLoss.forward\u001b[0;34m(self, input, target)\u001b[0m\n\u001b[1;32m   1184\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mforward\u001b[39m(\u001b[38;5;28mself\u001b[39m, \u001b[38;5;28minput\u001b[39m: Tensor, target: Tensor) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m Tensor:\n\u001b[0;32m-> 1185\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mF\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mcross_entropy\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43minput\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtarget\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mweight\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mweight\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m   1186\u001b[0m \u001b[43m                           \u001b[49m\u001b[43mignore_index\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mignore_index\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mreduction\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mreduction\u001b[49m\u001b[43m,\u001b[49m\n\u001b[1;32m   1187\u001b[0m \u001b[43m                           \u001b[49m\u001b[43mlabel_smoothing\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mlabel_smoothing\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/anaconda3/envs/PACFL/lib/python3.11/site-packages/torch/nn/functional.py:3086\u001b[0m, in \u001b[0;36mcross_entropy\u001b[0;34m(input, target, weight, size_average, ignore_index, reduce, reduction, label_smoothing)\u001b[0m\n\u001b[1;32m   3084\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m size_average \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m \u001b[38;5;129;01mor\u001b[39;00m reduce \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[1;32m   3085\u001b[0m     reduction \u001b[38;5;241m=\u001b[39m _Reduction\u001b[38;5;241m.\u001b[39mlegacy_get_string(size_average, reduce)\n\u001b[0;32m-> 3086\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mtorch\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_C\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_nn\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mcross_entropy_loss\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43minput\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mtarget\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mweight\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m_Reduction\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mget_enum\u001b[49m\u001b[43m(\u001b[49m\u001b[43mreduction\u001b[49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mignore_index\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mlabel_smoothing\u001b[49m\u001b[43m)\u001b[49m\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "h\n"
     ]
    }
   ],
   "source": [
    "print(\"h\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
