{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Partition Dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torchvision\n",
    "import torchvision.transforms as transforms\n",
    "from torch.utils.data import DataLoader, Subset\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "from scipy import stats\n",
    "import copy\n",
    "import scipy.io as sio"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def iid_divide(l, g):    \n",
    "    num_elems = len(l)\n",
    "    group_size = int(len(l) / g)\n",
    "    num_big_groups = num_elems - g * group_size\n",
    "    num_small_groups = g - num_big_groups\n",
    "    glist = []\n",
    "    for i in range(num_small_groups):\n",
    "        glist.append(l[group_size * i: group_size * (i + 1)])\n",
    "    bi = group_size * num_small_groups\n",
    "    group_size += 1\n",
    "    for i in range(num_big_groups):\n",
    "        glist.append(l[bi + group_size * i:bi + group_size * (i + 1)])\n",
    "    return glist"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def pathological_non_iid_split(dataset, n_classes, n_clients, n_classes_per_client,seed=2025):\n",
    "    np.random.seed(seed)\n",
    "    data_idcs = list(range(len(dataset)))\n",
    "    label2index = {k: [] for k in range(n_classes)}\n",
    "    for idx in data_idcs:\n",
    "        _, label = dataset[idx]\n",
    "        label2index[label].append(idx)\n",
    "\n",
    "    sorted_idcs = []\n",
    "    for label in label2index:\n",
    "        sorted_idcs += label2index[label]\n",
    "\n",
    "    n_shards = n_clients * n_classes_per_client\n",
    "    \n",
    "    shards = iid_divide(sorted_idcs, n_shards)\n",
    "    np.random.shuffle(shards)\n",
    "    \n",
    "    tasks_shards = iid_divide(shards, n_clients)\n",
    "\n",
    "    clients_idcs = [[] for _ in range(n_clients)]\n",
    "    for client_id in range(n_clients):\n",
    "        for shard in tasks_shards[client_id]:\n",
    "            clients_idcs[client_id] += shard\n",
    "    return clients_idcs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def iid_split(dataset, n_classes, n_clients, seed=2025):\n",
    "    np.random.seed(seed)\n",
    "    clients_idcs = []\n",
    "    train_labels = dataset.targets\n",
    "    class_idc = [np.argwhere(train_labels == y).flatten().tolist() for y in range(n_classes)]\n",
    "    class_idcs = []\n",
    "    for i in range(n_classes):\n",
    "        class_idcs += class_idc[i]    \n",
    "    class_idcs = np.random.permutation(class_idcs)\n",
    "\n",
    "    num_per_client = int(len(dataset) / n_clients)    \n",
    "    for i in range(n_clients):\n",
    "        client = []\n",
    "        for j in range(num_per_client):\n",
    "            client += [class_idcs[i*num_per_client + j]]\n",
    "        clients_idcs.append(client)\n",
    "\n",
    "    return clients_idcs\n",
    "\n",
    "# def iid_split(dataset, n_classes, n_clients, seed=2025):\n",
    "#     np.random.seed(seed)\n",
    "#     perm = np.random.permutation(len(dataset))\n",
    "    \n",
    "    \n",
    "    \n",
    "#     clients_idcs = []\n",
    "#     train_labels = dataset.targets\n",
    "#     class_idc = [np.argwhere(train_labels == y).flatten().tolist() for y in range(n_classes)]\n",
    "#     class_idcs = []\n",
    "#     for i in range(n_classes):\n",
    "#         class_idcs += class_idc[i]    \n",
    "#     class_idcs = np.random.permutation(class_idcs)\n",
    "\n",
    "#     num_per_client = int(len(dataset) / n_clients)    \n",
    "#     for i in range(n_clients):\n",
    "#         client = []\n",
    "#         for j in range(num_per_client):\n",
    "#             client += [class_idcs[i*num_per_client + j]]\n",
    "#         clients_idcs.append(client)\n",
    "\n",
    "    return clients_idcs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "transform = transforms.Compose([\n",
    "    transforms.ToTensor(),\n",
    "    transforms.Normalize((0.1307,), (0.3081,))\n",
    "])\n",
    "trainset = torchvision.datasets.MNIST(root='./data', train=True, download=True, transform=transform)\n",
    "# trainset = torchvision.datasets.MNIST(root='./data', train=True, download=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "The PostScript backend does not support transparency; partially transparent artists will be rendered opaque.\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 600x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "n_classes = 10\n",
    "n_clients = 60\n",
    "classes = trainset.classes\n",
    "n_classes = len(classes)\n",
    "labels = trainset.targets\n",
    "# seed = 123\n",
    "seed = 2025\n",
    "np.random.seed(seed)\n",
    "num_per_client = int(len(trainset) / n_clients)\n",
    "clients_idc = np.random.permutation(len(trainset))\n",
    "clients_idcs=[]\n",
    "for i in range(n_clients):\n",
    "    clients_idcs.append(clients_idc[i*num_per_client : (i+1)*num_per_client])\n",
    "# clients_idcs = iid_split(trainset, n_classes, n_clients,seed)\n",
    "\n",
    "\n",
    "\n",
    "plt.figure(figsize=(6, 4))\n",
    "label_distribution = [[] for _ in range(n_classes)]\n",
    "for c_id, idc in enumerate(clients_idcs):\n",
    "    for idx in idc:\n",
    "        label_distribution[labels[idx]].append(c_id)\n",
    "\n",
    "plt.hist(label_distribution, stacked=True,\n",
    "            bins=np.arange(-0.5, n_clients + 1.5, 1),\n",
    "            label=range(n_classes), rwidth=0.5)\n",
    "# plt.xticks(np.arange(n_clients), [\"%d\" %\n",
    "                                    # (c_id+1) for c_id in range(n_clients)], fontsize=8)\n",
    "# plt.xlabel(\"Client ID\",fontsize=15)\n",
    "# plt.ylabel(\"Number of samples\",fontsize=15)\n",
    "plt.legend(loc=\"upper right\")\n",
    "plt.title(\"Display Label Distribution (I.I.D.)\")\n",
    "plt.savefig('MNIST_distr_iid.eps', format=\"eps\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "clients = []\n",
    "X = np.zeros((60000, 784))\n",
    "ncostterms = int(60000/n_clients)\n",
    "for i in range(n_clients):    \n",
    "    for j in range(ncostterms):\n",
    "        x, y = trainset[clients_idcs[i][j]]\n",
    "        x = x.flatten().tolist()\n",
    "        X[i*ncostterms+j, :] = x\n",
    "sio.savemat(f'MNIST_train_{n_clients}_iid.mat', {'data':X, 'nagents': n_clients, 'ncostterms': ncostterms})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([-0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "        0.39039403,  2.82148671,  2.82148671,  2.00687981,  1.20500112,\n",
       "        1.20500112, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296,  1.20500112,  2.82148671,\n",
       "        2.82148671,  2.82148671,  2.82148671,  2.82148671,  2.00687981,\n",
       "        0.39039403, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296,  1.20500112,  2.82148671,  2.82148671,  2.82148671,\n",
       "        2.82148671,  2.82148671,  2.82148671,  2.82148671,  0.39039403,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296,  1.20500112,\n",
       "        2.82148671,  2.82148671,  1.20500112, -0.42421296,  2.82148671,\n",
       "        2.82148671,  2.82148671,  2.82148671, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296,  1.20500112,  2.82148671,  2.82148671,\n",
       "        1.20500112, -0.42421296, -0.42421296,  2.00687981,  2.82148671,\n",
       "        2.82148671,  1.20500112, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "        1.20500112,  2.82148671,  2.82148671,  2.00687981,  1.20500112,\n",
       "       -0.42421296,  1.20500112,  2.82148671,  2.82148671,  1.20500112,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296,  1.20500112,  2.82148671,\n",
       "        2.82148671,  2.82148671,  2.82148671,  2.82148671,  2.82148671,\n",
       "        2.82148671,  2.82148671,  2.82148671,  1.20500112, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296,  1.20500112,  2.82148671,  2.82148671,\n",
       "        2.82148671,  2.82148671,  2.82148671,  2.82148671,  2.82148671,\n",
       "        2.82148671,  2.00687981, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296,  1.20500112,  2.82148671,  2.82148671,  2.82148671,\n",
       "        2.82148671,  2.82148671,  2.82148671,  2.82148671,  1.20500112,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296,  0.39039403,  1.20500112,  0.39039403,  1.20500112,\n",
       "        2.82148671,  2.82148671,  1.20500112, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296,  2.00687981,  2.82148671,  2.82148671,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "        2.82148671,  2.82148671,  2.82148671, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296,  1.20500112,  2.82148671,  2.82148671,\n",
       "        1.20500112, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296,  0.39039403,\n",
       "        2.82148671,  2.82148671,  2.82148671,  0.39039403, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296,  0.39039403,  2.82148671,  2.82148671,  2.82148671,\n",
       "        2.00687981, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296,  2.00687981,  2.82148671,\n",
       "        2.82148671,  2.82148671,  2.00687981, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "        0.39039403,  1.20500112,  1.20500112,  1.20500112,  1.20500112,\n",
       "        2.82148671,  2.82148671,  2.82148671,  2.82148671,  2.00687981,\n",
       "        0.39039403, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296,  2.00687981,  2.82148671,\n",
       "        2.82148671,  2.82148671,  2.82148671,  2.82148671,  2.82148671,\n",
       "        2.82148671,  1.20500112,  0.39039403, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296,  2.82148671,  2.82148671,  2.82148671,  2.82148671,\n",
       "        2.82148671,  2.82148671,  2.00687981,  0.39039403, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296,  1.20500112,\n",
       "        2.82148671,  2.82148671,  1.20500112, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296, -0.42421296,\n",
       "       -0.42421296, -0.42421296, -0.42421296, -0.42421296])"
      ]
     },
     "execution_count": 42,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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1KlT57x1uNOSk5PP++ycnBwz+FeJAACA2CkyP8A1evRo+dOf/hTzz1m2/Erf69Q2ewKmabuNUK8vNM1rnbbTUlZs9b0+2LppVKbZbrvXOm2Xi0VavKb985lPfa8vH3NrVNYZi+/hYq/TK19ise/GYrlof7f+70drmn/6g7fBa5rXecj287zW6bWc7bZPenx5wLS0KW1i+nm2+ZLisVws8qygiGp1RkpKivl/6NChgPd13J2m/w8fPhww/cyZM6bHhv88odbh/xnBhg4dKllZWb5BfwIcAAAUkiBCqyD0Ir9s2bKAagVt69CyZUszrv8zMzNNrwvX8uXLJTc317SdcOfRHhunT5/2zaM9OerVqxeyKkMlJib6fvabn/8GAKAABhH6PIetW7eawW1Mqa/37t1rnhvRv39/GTVqlHz44YeSnp4uv//9780zJTp37mzmb9Cggdx1113Sq1cv2bBhg3z++efSt29f6dq1q5lP/e53vzONKvX5EdoVdM6cOfL666/LwIEDo739AADgYrWJ2LRpk7Ru3do37l7Ye/ToIdOmTZOnn37aPEtCn/ugJQ6tWrWSxYsXm4dGuWbMmGECh9TUVNMro0uXLubZEv49OpYsWSJpaWnSrFkzueSSS8wDrPyfJQEAAApZEHHHHXeYp1KGo6URI0eONEM42hNj5syZnp9z7bXXyqefnmvwBAAAChZ+OwMAAFghiAAAAFYIIgAAgBWCCAAAYIUgAgAAWCGIAAAAVggiAACAFYIIAABghSACAABYIYgAAABWCCIAAIAVgggAAGCFIAIAAFghiAAAAFYIIgAAgBWCCAAAYIUgAgAAWCGIAAAAVggiAACAFYIIAABghSACAABYIYgAAABWCCIAAIAVgggAAGCFIAIAAFghiAAAAFYIIgAAgBWCCAAAYIUgAgAAWCGIAAAAVggiAACAlQS7xZBXj2anhp326eruvtepbSKf5mWG08VvbI/kJ6+0eOWLl61vNgibL+V3bAq73OOr5p8bad004nQuyDzte50WtE6v5Ww/z2tavSXTzo0EbfumT477fZ5EzHY/m5Pxku/1ILlVYs3re/fKa6/t89oG2+/PRuD6Il+nf54Eb5/t8RWwj6k2kR1f/vkcnBYvAfkclNdtVgYfcTt+8efZ5rXX580oQOu8mCiJAAAAVggiAACAFYIIAABghSACAABY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8efJtZgo0p8njzcZGRnmwWD+eaTPXtdqn3jLI/en6CtXrmz+676jpRP+eaNFtbVq1YqbvDl79qzMnj3blM5otUa854mWXHXs2DFg+1U854sWwWtV6a9+9Svp1q2b7N27V+I9Tz788EPz5OYHHnjAVJVed9118vbbbxeY8y5BRIT+9a9/mZNg8BMzdVy/QIgvH+I9j/R3XrR+W4sgGzVqZN7T7dffhAn+Ubh4yJv09HRTh61P1Xv88cdl3rx50rBhw7jOEw2mtEpU29IEi9d80YvetGnTzJOJtS2NXhxvvfVW80uS8Zon6ttvvzX5UbduXfnkk0+kT58+8tRTT8n06dMLxHm3wD72GijMd5hff/11QH1uPKtXr55s3brVlM787W9/M7+Jo3Xa8Up/trlfv36m7Yw20Ma/dejQwfda24hoUFG7dm15//33TWPBeJWbm2tKIl588UUzriURen7R9g96LOU3SiJi+PPk8cbNh3jOo759+8qiRYtkxYoVplGhS7dfq8QyMzPjLm/0DvKqq64yv8yrd97aMPf111+P2zzRonltjH399ddLQkKCGTSo0oZx+lrvIOMxX4JpqcPVV18t33zzTdzuK0p7XGjJnb8GDRr4qnry+7xLEBEhfp78wurUqWN2Wv88OnbsmGktXNTzSNuZagChRfXLly83eeFP9x1tXe2fN9oFVE8ERT1vgulxk5OTE7d5kpqaaqp4tHTGHfROU9sAuK/jMV+CnThxQvbs2WMuovG6ryitFg3uLr5r1y5TSlMgzrsxb7pZhMyePdu0eJ02bZqzfft2p3fv3k6lSpWcgwcPOvFCW5R/8cUXZtDdZ/z48eb1999/b6aPGTPG5MmCBQucr776yunUqZNTp04d5+eff3aKsj59+jgVK1Z0Vq5c6fzwww++4aeffvLN8/jjjzu1atVyli9f7mzatMlp2bKlGYqyZ555xvRQycjIMPuDjhcrVsxZsmRJ3OZJKP69M+I1XwYNGmSOH91XPv/8c6dt27bOJZdcYno6xWueqA0bNjgJCQnOCy+84OzevduZMWOGU6ZMGee9995zXPl53iWIyKOJEyeaHblUqVKmy+e6deuceLJixQoTPAQPPXr08HU3evbZZ51q1aqZgCs1NdXZuXOnU9SFyhMdpk6d6ptHD+gnnnjCdHHUk8B9991nAo2i7JFHHnFq165tjpdLL73U7A9uABGveRJJEBGP+fLQQw851atXN/vKZZddZsa/+eabuM4T18KFC51GjRqZc2r9+vWdt956y/GXn+ddfgocAABYoU0EAACwQhABAACsEEQAAAArBBEAAMAKQQQAALBCEAEAAKwQRAAAACsEEQAAwApBBAAAsEIQAQAArBBEAAAAKwQRAABAbPw/3KSc9scnEKAAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 600x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "n_clients = 60\n",
    "# n_classes_per_client = 4  # heavy non-i.i.d.\n",
    "n_classes_per_client = 8  # slight non-i.i.d.\n",
    "seed = 2025\n",
    "\n",
    "classes = trainset.classes\n",
    "n_classes = len(classes)\n",
    "\n",
    "# labels = np.concatenate(\n",
    "#     [np.array(trainset.targets), np.array(testset.targets)], axis=0)\n",
    "# dataset = ConcatDataset([trainset, testset])\n",
    "\n",
    "labels = trainset.targets\n",
    "\n",
    "# \n",
    "clients_idcs = pathological_non_iid_split(\n",
    "    trainset, n_classes, n_clients, n_classes_per_client,seed)\n",
    "\n",
    "# show local dataset distributions\n",
    "plt.figure(figsize=(6, 4))\n",
    "label_distribution = [[] for _ in range(n_classes)]\n",
    "for c_id, idc in enumerate(clients_idcs):\n",
    "    for idx in idc:\n",
    "        label_distribution[labels[idx]].append(c_id)\n",
    "\n",
    "plt.hist(label_distribution, stacked=True,\n",
    "            bins=np.arange(-0.5, n_clients + 1.5, 1),\n",
    "            label=range(n_classes), rwidth=0.5)\n",
    "# plt.xticks(np.arange(n_clients), [\"%d\" %\n",
    "                                    # (c_id+1) for c_id in range(n_clients)], fontsize=8)\n",
    "# plt.xlabel(\"Client ID\",fontsize=15)\n",
    "# plt.ylabel(\"Number of samples\",fontsize=15)\n",
    "# plt.legend(loc=\"upper right\")\n",
    "plt.title(\"Display Label Distribution (Non-I.I.D., slight)\")\n",
    "# plt.title(\"Display Label Distribution (Non-I.I.D., heavy)\")\n",
    "plt.savefig('MNIST_distr_noniid_slight.eps', format=\"eps\")\n",
    "# plt.savefig('MNIST_distr_noniid_heavy.eps', format=\"eps\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 124,
   "metadata": {},
   "outputs": [],
   "source": [
    "clients = []\n",
    "X = np.zeros((60000, 784))\n",
    "ncostterms = int(60000/n_clients)\n",
    "for i in range(n_clients):    \n",
    "    for j in range(len(clients_idcs[i])):\n",
    "        x, y = trainset[clients_idcs[i][j]]\n",
    "        x = np.array(x.flatten().tolist())\n",
    "        X[i*ncostterms+j, :] = x\n",
    "sio.savemat(f'MNIST_train_{n_clients}_noniid_large.mat', {'data':X, 'nagents': n_clients, 'ncostterms': ncostterms})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 115,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[1, 2]"
      ]
     },
     "execution_count": 115,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "FLdataset",
   "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.13.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
