{
 "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",
    "\n",
    "            clients_idcs[client_id] += shard\n",
    "    return clients_idcs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "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": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 170M/170M [00:19<00:00, 8.68MB/s] \n"
     ]
    }
   ],
   "source": [
    "transform = transforms.Compose([\n",
    "    transforms.ToTensor(),  \n",
    "    transforms.Normalize((0.4914, 0.4822, 0.4465),   \n",
    "                         (0.2023, 0.1994, 0.2010))   \n",
    "])\n",
    "\n",
    "trainset = torchvision.datasets.CIFAR10(root='./data', train=True,\n",
    "                                        download=True, transform=transform)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## IID"
   ]
  },
  {
   "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": "iVBORw0KGgoAAAANSUhEUgAAAhEAAAF2CAYAAADQh8ptAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjEsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvc2/+5QAAAAlwSFlzAAAPYQAAD2EBqD+naQAAOxNJREFUeJzt3Qd8FGX++PFvqKGFKgSkCkiRptgQRSmCyHmAnO1QI3LoQVDKHSL+EAFBED2kiFiOAxsgqEEPlS4g0lEURFE4BJRA8IdJCCWU7P/1fX7/3dtdUjbDzu7O7uf9ei3LlOw88+zszHeeNnEul8slAAAAhVSksH8AAACgCCIAAIAlBBEAAMASgggAAGAJQQQAALCEIAIAAFhCEAEAACwhiAAAAJYQRAAAAEsIIhCRRo8eLXFxcbZ9/kMPPSR169aVaMij3377zRH5op+rn2+3n3/+2eTLnDlzPPN0u2XLlpVQ0e3r9xMuAwYMkFtvvVWc4Prrr5cnnngi3MmARQQRsJ2ezPWk6n7Fx8dLjRo1pEuXLjJt2jQ5fvy4xIpbbrlFmjVrJtGwH+7vs0iRIpKQkCCNGjWSBx54QJYvXx607Xz66adhvRg7MW379u2Tf/7zn/LUU09dEFi9+OKLBf69rjdw4MCAgkLvY6BChQrSvHlzeeSRR2TTpk0Bp3f48OEyY8YMOXz4cMB/g8hRLNwJQOwYO3as1KtXT86ePWtOGKtXr5bBgwfL5MmT5eOPP5YWLVp41h05cqQ8+eSTYU0v8lezZk2ZMGGC+f+JEydkz5498uGHH8o777wjd999t3kvXry4Z/3du3ebi01hL9R6gSnMxbpOnTpy6tQpn23bIb+06faLFQvP6XXq1Knmd9a+fXvbt9WqVSv529/+Zv6vNwPff/+9LFy4UN544w0ZMmSI+W0XpHv37iYIfeWVV8w5As5CEIGQ6dq1q1x99dWe6REjRsiqVavkD3/4g/zxj380J6BSpUqZZXoCDtdJGIEpX7683H///T7zJk6cKI8//ri5IOid6vPPP+9ZVrJkSVvTc+7cOcnJyZESJUqY0q5wCtf2NUB/99135a9//WtItnfppZdecAzod/7nP/9ZXnrpJWnYsKH0798/38/QwPJPf/qTvPXWWzJmzBhbqzERfFRnIKw6dOggTz/9tOzfv9/cuebXJkKLyW+88UZTbKr121p87l1kqyUb+jfvvfeemZ+YmChlypQxAcrBgwcLTIsW9d5www1SuXJlE8y0bt1a3n//fZ91br75ZmnZsmWuf6/p0Sqai/Xtt9+aOvzLLrvMXIx0Px5++GH53//931zX1zYReuevd3Oa9kGDBsnp06cvWE/zV/dJ961SpUpy7733BpQvhVG0aFFTRdW0aVN5+eWXJSMjI882EXrB04uGXmh0PzXt+v26q0N0Xb3TV97VYf7F81OmTJH69eubIGXXrl25tolw+89//mO+Iz0utEpN73y9H2TsPob03Zv/Z+aXNvc8/xKKr7/+2gTS+j3p8duxY0fZuHFjrlV/X375pQwdOlQuueQSk9aePXvK0aNHC8z/devWmeOhU6dOEi56fL399tvmGBs/frxP/uZF22/oOWD79u0hSSOChyACYaf16GrZsmV5rvPdd9+ZEovs7Gxz4v/HP/5hggM92frTE9cnn3xi6lr1rlgvSnpS1SLmgoqBr7zySvP5zz33nCkJueuuu8xneadVL/I7d+70+dstW7bIjz/+eMFdmRWaXr3Y9enTR6ZPn24u9vPnz5fbb7891xOyBhAaNGjVgq6jF3Gtl/bPkwcffNBcsLWIWauRVq5cKe3atZP09HQJdiBx3333ycmTJ81FLS96kdUgQovdNeD4n//5H6ldu7Z89dVXZvmjjz7qaRyoFyX3y9vs2bNNHun+6jGhF668nD9/Xm677TapVq2aTJo0yQRUzzzzjHkVViBp8z9+b7rpJvnmm29MI0INnLXtgrYtya39wGOPPWbW1bTpnfy///3vgNoprF+/3gQhehyHkwZJGvj8+uuvJrAriH4XKrffMyKcC7DZ7Nmz9crn2rJlS57rlC9f3nXllVd6pp955hnzN24vvfSSmT569Gien/H555+bdS699FJXZmamZ/6CBQvM/KlTp3rmJSUluerUqePz9ydPnvSZPnPmjKtZs2auDh06eOalp6e74uPjXcOHD/dZ9/HHH3eVKVPGlZWVlU9OuFw333yz64orrsh3Hf90qHnz5pl9WLt27QV59Mc//tFn3QEDBpj533zzjZn++eefXUWLFnWNHz/eZ70dO3a4ihUr5jM/t3yxsh8pKSkX5Ll+rn6+W8uWLV3dunXLdzvJyck+x4Hbvn37zPyEhARXWlparsv0uPPeL5332GOPeebl5OSY7Zco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",
      "text/plain": [
       "<Figure size 600x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "n_classes = 10\n",
    "n_clients = 50\n",
    "classes = trainset.classes\n",
    "n_classes = len(classes)\n",
    "labels = trainset.targets\n",
    "\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('CIFAR10_distr_iid.eps', format=\"eps\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "clients = []\n",
    "X = np.zeros((50000, 3072))\n",
    "ncostterms = int(50000/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'CIFAR10_train_{n_clients}_iid.mat', {'data':X, 'nagents': n_clients, 'ncostterms': ncostterms})"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## NON-IID"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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1JzPlB4jbjnV496ladbgKti5tMRzUCzMmjVLr3zO1q1bc92mWrVqznXXXeebf/rpp81rXC+//LKZ//HHH3Pdx+rVq802l112mXP8+HHf8nfeeccsf/XVV33L+vbt6zRo0MDz+p9//tkzf+bMGad58+ZOcnKyb1lmZqYTHx/vPPHEE55tH3vsMadSpUrOyZMn88gJx7ntttuca665Js9tAtOh5s2bZz7DunXrcuTRb37zG8+2gwYNMss/++wzM//tt986ZcqUcZ577jnPdunp6U7ZsmU9y4Pli83nWLRoUY481/3q/l0tW7Z0unbtmuf7pKamesqB68CBA2Z51apVnaNHjwZdp+XO/3PpskcffdS37MKFC+b9y5cv7ytXbhnSv6H2mVvalC7X78fVvXt38z779+/3LTt8+LBTpUoVp3379jmOlQ4dOpj0uYYNG2a+Qy1/eXnggQecmjVr5ppfkyZNcs6dO+c0atTI5L/7Hm5ZcvNBy36tWrVM+f/ll198+1m6dKnZbvTo0TnyduzYsZ731OO5devWeaY3MG2h6Haa76FoWcurbLnnkyVLljjR0K1bt5DHtfvd6uf1P450cr300ktmm8WLF/uWaf43adIkR7nU1+myt99+27csOzvbSUxMdHr06OFbpufdwLIb6Oqrr3Y6d+7sXGyoiShB9O4gr1Ea2oShlixZYtq986J33FWqVPHN//a3v5U6deqYjnB5cftjuBG+VsXr3aN7Z+z2A+jWrZvMmzfPVyugd2HahKJVuVoFmV/+6dAaBq2e1s5gyj8trtTUVM+822nV/bx696h5pneVui930mYSrZlYvXq1RJvbthrqO9U79H379lm/T48ePUzVebj87+bdnv5nzpwxd34FRcuH1rJp+dAmKpeWSW2f19qa48ePe16jd43+zSNaDnU/WtWdF23y0qr0vPjXRixevDjoNtu2bZOjR4+aoaL+/Tu6du1q7nb9a+dcgf0wNM1aoxaLwimfkdCy/K9//cvUSObHsmXLTJOE1rK6NP+1Fii3z+Ff+6l9LG688caI8z0hIaFIRuYUNYKIEuTkyZOeC38g7e1/yy23yEMPPWSqo7WK/5133gkaUOiF0Z+ejK+66irP+Oxgli5dai7WetBqtbhenHQkiX+7vhukHDx4UNavX2/m9QJ05MiRqI0GOHbsmOnXoJ9TAwpNh/a2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",
      "text/plain": [
       "<Figure size 600x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "n_clients = 50\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",
    "\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('CIFAR10_distr_noniid_slight.eps', format=\"eps\")\n",
    "# plt.savefig('CIFAR10_distr_noniid_heavy.eps', format=\"eps\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [],
   "source": [
    "clients = []\n",
    "X = np.zeros((50000, 3072))\n",
    "ncostterms = int(50000/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'CIFAR10_train_{n_clients}_noniid_slight.mat', {'data':X, 'nagents': n_clients, 'ncostterms': ncostterms})\n",
    "# sio.savemat(f'CIFAR10_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
}
