{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "a937404e-c65a-4c44-854c-6663deef0729",
   "metadata": {},
   "outputs": [],
   "source": [
    "from PIL import Image\n",
    "import os\n",
    "import numpy as np\n",
    "import torch\n",
    "from torchvision import datasets, transforms\n",
    "from util.sampling import iid_sampling, non_iid_dirichlet_sampling\n",
    "import torch.utils\n",
    "import sys\n",
    "\n",
    "import torch.multiprocessing as mp\n",
    "mp.set_sharing_strategy('file_system')\n",
    "\n",
    "# setting path\n",
    "sys.path.append('../')\n",
    "\n",
    "import math\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torchvision.transforms import Compose, Normalize, RandomVerticalFlip\n",
    "from gossipy_original.core import AntiEntropyProtocol, CreateModelMode, StaticP2PNetwork\n",
    "from gossipy_original.data import DataDispatcher\n",
    "\n",
    "from gossipy_original.model import TorchModel\n",
    "from gossipy_original.data.handler import ClassificationDataHandler\n",
    "from gossipy_original.model.handler import TorchModelHandler\n",
    "from gossipy_original.node import PENSNode\n",
    "from gossipy_original.simul import GossipSimulator, SimulationReport\n",
    "from gossipy_original.data import get_CIFAR10, get_CIFAR100\n",
    "from gossipy_original.utils import plot_evaluation\n",
    "from torchvision import models\n",
    "\n",
    "# from models import *\n",
    "import os\n",
    "import torch\n",
    "import torch.nn.functional as F\n",
    "from torch.autograd import Variable\n",
    "from data.datasets import input_dataset\n",
    "import argparse\n",
    "import torchvision.transforms as transforms\n",
    "from torch.utils.data import DataLoader, random_split\n",
    "from torchvision.models import resnet34\n",
    "import os\n",
    "import torch\n",
    "import datetime\n",
    "# import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "import pandas as pd\n",
    "import scipy as sp\n",
    "import cv2\n",
    "import numpy as np\n",
    "from torchvision import datasets, transforms\n",
    "from torch.utils.data import DataLoader\n",
    "from torch import nn, optim\n",
    "from PIL import ImageFile\n",
    "    \n",
    "\n",
    "from PIL import Image\n",
    "import os\n",
    "import numpy as np\n",
    "import torch\n",
    "from torchvision import datasets, transforms\n",
    "from util.sampling import iid_sampling, non_iid_dirichlet_sampling\n",
    "import torch.utils\n",
    "import sys\n",
    "\n",
    "import torch.multiprocessing as mp\n",
    "mp.set_sharing_strategy('file_system')\n",
    "\n",
    "# setting path\n",
    "sys.path.append('../')\n",
    "\n",
    "import math\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torchvision.transforms import Compose, Normalize, RandomVerticalFlip\n",
    "from gossipy_original.core import AntiEntropyProtocol, CreateModelMode, StaticP2PNetwork\n",
    "from gossipy_original.data import DataDispatcher\n",
    "\n",
    "from gossipy_original.model import TorchModel\n",
    "from gossipy_original.data.handler import ClassificationDataHandler\n",
    "from gossipy_original.model.handler import TorchModelHandler\n",
    "from gossipy_original.node import PENSNode\n",
    "from gossipy_original.simul import GossipSimulator, SimulationReport\n",
    "from gossipy_original.data import get_CIFAR10, get_CIFAR100\n",
    "from gossipy_original.utils import plot_evaluation\n",
    "from torchvision import models\n",
    "\n",
    "# from models import *\n",
    "import os\n",
    "import torch\n",
    "import torch.nn.functional as F\n",
    "from torch.autograd import Variable\n",
    "from data.datasets import input_dataset\n",
    "import argparse\n",
    "import torchvision.transforms as transforms\n",
    "from torch.utils.data import DataLoader, random_split\n",
    "from torchvision.models import resnet34\n",
    "import os\n",
    "import torch\n",
    "import datetime\n",
    "# import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "import pandas as pd\n",
    "import scipy as sp\n",
    "import cv2\n",
    "import numpy as np\n",
    "from torchvision import datasets, transforms\n",
    "from torch.utils.data import DataLoader\n",
    "from torch import nn, optim\n",
    "from PIL import ImageFile\n",
    "    \n",
    "\n",
    "class CustomDataDispatcher(DataDispatcher):\n",
    "    def assign(self, seed: int = 42) -> None:\n",
    "        self.tr_assignments = [[] for _ in range(self.n)]\n",
    "        self.te_assignments = [[] for _ in range(self.n)]\n",
    "\n",
    "        n_ex = self.data_handler.size()\n",
    "        ex_x_user = math.ceil(n_ex / self.n)\n",
    "\n",
    "        for idx, i in enumerate(range(0, n_ex, ex_x_user)):\n",
    "            self.tr_assignments[idx] = list(range(i, min(i + ex_x_user, n_ex)))\n",
    "\n",
    "        if self.eval_on_user:\n",
    "            n_eval_ex = self.data_handler.eval_size()\n",
    "            eval_ex_x_user = math.ceil(n_eval_ex / self.n)\n",
    "            for idx, i in enumerate(range(0, n_eval_ex, eval_ex_x_user)):\n",
    "                self.te_assignments[idx] = list(range(i, min(i + eval_ex_x_user, n_eval_ex)))\n",
    "import numpy as np\n",
    "import torch\n",
    "from torchvision import datasets, transforms\n",
    "from PIL import Image\n",
    "import os\n",
    "import math\n",
    "\n",
    "class Clothing(torch.utils.data.Dataset):\n",
    "    def __init__(self, root, transform, mode):\n",
    "        self.root = root\n",
    "        self.noisy_labels = {}\n",
    "        self.clean_labels = {}\n",
    "        self.data = []\n",
    "        self.targets = []\n",
    "        self.transform = transform\n",
    "        self.mode = mode\n",
    "\n",
    "        with open(self.root + 'noisy_label_kv.txt', 'r') as f:\n",
    "            lines = f.read().splitlines()\n",
    "        for l in lines:\n",
    "            entry = l.split()\n",
    "            img_path = self.root + entry[0]\n",
    "            self.noisy_labels[img_path] = int(entry[1])\n",
    "\n",
    "        with open(self.root + 'clean_label_kv.txt', 'r') as f:\n",
    "            lines = f.read().splitlines()\n",
    "        for l in lines:\n",
    "            entry = l.split()\n",
    "            img_path = self.root + entry[0]\n",
    "            self.clean_labels[img_path] = int(entry[1])\n",
    "\n",
    "        if self.mode == 'train':\n",
    "            with open(self.root + 'noisy_train_key_list.txt', 'r') as f:\n",
    "                lines = f.read().splitlines()\n",
    "            for l in lines:\n",
    "                img_path = self.root + l\n",
    "                self.data.append(img_path)\n",
    "                target = self.noisy_labels[img_path]\n",
    "                self.targets.append(target)\n",
    "        elif self.mode == 'minitrain':\n",
    "            with open(self.root + 'noisy_train_key_list.txt', 'r') as f:\n",
    "                lines = f.read().splitlines()\n",
    "            n = len(lines)\n",
    "            np.random.seed(13)\n",
    "            subset_idx = np.random.choice(n, int(n/10), replace=False)\n",
    "            for i in subset_idx:\n",
    "                l = lines[i]\n",
    "                img_path = self.root + l\n",
    "                self.data.append(img_path)\n",
    "                target = self.noisy_labels[img_path]\n",
    "                self.targets.append(target)\n",
    "        elif self.mode == 'test':\n",
    "            with open(self.root + 'clean_test_key_list.txt', 'r') as f:\n",
    "                lines = f.read().splitlines()\n",
    "            for l in lines:\n",
    "                img_path = self.root + l\n",
    "                self.data.append(img_path)\n",
    "                target = self.clean_labels[img_path]\n",
    "                self.targets.append(target)\n",
    "\n",
    "    def __getitem__(self, index):\n",
    "        img_path = self.data[index]\n",
    "        target = self.targets[index]\n",
    "        image = Image.open(img_path).convert('RGB')\n",
    "        img = self.transform(image)\n",
    "        return img, target\n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self.data)\n",
    "\n",
    "def get_dataset(args):\n",
    "    args.device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "\n",
    "    if args.dataset == 'cifar10':\n",
    "        data_path = '../data/cifar10'\n",
    "        args.num_classes = 10\n",
    "        trans_train = transforms.Compose([\n",
    "            transforms.RandomCrop(32, padding=4),\n",
    "            transforms.RandomHorizontalFlip(),\n",
    "            transforms.ToTensor(),\n",
    "            transforms.Normalize(mean=[0.485, 0.456, 0.406],\n",
    "                                 std=[0.229, 0.224, 0.225])],\n",
    "        )\n",
    "        trans_val = transforms.Compose([\n",
    "            transforms.ToTensor(),\n",
    "            transforms.Normalize(mean=[0.485, 0.456, 0.406],\n",
    "                                 std=[0.229, 0.224, 0.225])],\n",
    "        )\n",
    "        dataset_train = datasets.CIFAR10(data_path, train=True, download=True, transform=trans_train)\n",
    "        dataset_test = datasets.CIFAR10(data_path, train=False, download=True, transform=trans_val)\n",
    "        n_train = len(dataset_train)\n",
    "        y_train = np.array(dataset_train.targets)\n",
    "    elif args.dataset == 'cifar100':\n",
    "        data_path = '../data/cifar100'\n",
    "        args.num_classes = 100\n",
    "        args.model = 'resnet34'\n",
    "        trans_train = transforms.Compose([\n",
    "            transforms.RandomCrop(32, padding=4),\n",
    "            transforms.RandomHorizontalFlip(),\n",
    "            transforms.ToTensor(),\n",
    "            transforms.Normalize(mean=[0.507, 0.487, 0.441],\n",
    "                                 std=[0.267, 0.256, 0.276])],\n",
    "        )\n",
    "        trans_val = transforms.Compose([\n",
    "            transforms.ToTensor(),\n",
    "            transforms.Normalize(mean=[0.507, 0.487, 0.441],\n",
    "                                 std=[0.267, 0.256, 0.276])],\n",
    "        )\n",
    "        dataset_train = datasets.CIFAR100(data_path, train=True, download=True, transform=trans_train)\n",
    "        dataset_test = datasets.CIFAR100(data_path, train=False, download=True, transform=trans_val)\n",
    "        n_train = len(dataset_train)\n",
    "        y_train = np.array(dataset_train.targets)\n",
    "\n",
    "    elif args.dataset == 'clothing1m':\n",
    "        data_path = os.path.abspath('..') + '/Cloth1M/data/clothing1M/'\n",
    "        args.num_classes = 14\n",
    "        args.model = 'resnet50'\n",
    "        trans_train = transforms.Compose([\n",
    "                    transforms.Resize((256, 256)),\n",
    "                    transforms.RandomCrop(224),\n",
    "                    transforms.RandomHorizontalFlip(),\n",
    "                    transforms.ToTensor(),\n",
    "                    transforms.Normalize(mean=[0.485, 0.456, 0.406], std=[0.229, 0.224, 0.225]),\n",
    "                 ])\n",
    "        trans_val = transforms.Compose([\n",
    "                    transforms.Resize((224, 224)),\n",
    "                    transforms.ToTensor(),\n",
    "                    transforms.Normalize(mean=[0.485, 0.456, 0.406], std=[0.229, 0.224, 0.225]),\n",
    "                 ])\n",
    "        dataset_train = Clothing(data_path, trans_train, \"train\")\n",
    "        dataset_test = Clothing(data_path, trans_val, \"test\")\n",
    "\n",
    "        # Shuffle and slice the datasets\n",
    "        indices_train = np.arange(len(dataset_train.data))\n",
    "        np.random.shuffle(indices_train)\n",
    "        dataset_train.data = [dataset_train.data[i] for i in indices_train]\n",
    "        dataset_train.targets = [dataset_train.targets[i] for i in indices_train]\n",
    "\n",
    "        indices_test = np.arange(len(dataset_test.data))\n",
    "        np.random.shuffle(indices_test)\n",
    "        dataset_test.data = [dataset_test.data[i] for i in indices_test]\n",
    "        dataset_test.targets = [dataset_test.targets[i] for i in indices_test[:10000]]\n",
    "\n",
    "        n_train = len(dataset_train)\n",
    "        y_train = np.array(dataset_train.targets)\n",
    "\n",
    "    else:\n",
    "        exit('Error: unrecognized dataset')\n",
    "\n",
    "    if args.iid:\n",
    "        dict_users = iid_sampling(n_train, args.num_users, args.seed)\n",
    "    else:\n",
    "        dict_users = non_iid_dirichlet_sampling(y_train, args.num_classes, args.non_iid_prob_class, args.num_users, args.seed, args.alpha_dirichlet)\n",
    "\n",
    "    return dataset_train, dataset_test, dict_users\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "4a6b786b-51aa-412f-bb04-3d9a77c34531",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "from torchvision import models\n",
    "# from getdataset import get_dataset\n",
    "\n",
    "# Define arguments\n",
    "class Args:\n",
    "    def __init__(self):\n",
    "        self.dataset = 'clothing1m'\n",
    "        self.iid = True\n",
    "        self.num_users = 1\n",
    "        self.seed = 42\n",
    "        self.non_iid_prob_class = 0.5\n",
    "        self.alpha_dirichlet = 0.5\n",
    "\n",
    "args = Args()\n",
    "\n",
    "# Get the dataset\n",
    "train_dataset, test_dataset, _ = get_dataset(args)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "beed7358-e1ad-4c49-abdf-bd8a582431d6",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "10000"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(test_dataset)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "a433b581-7eea-443f-b527-2df6222284d9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "60000"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(train_dataset)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "67f02a07-6dc6-46c5-9262-430b2eba5b27",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "db91ef61-e37a-425a-9184-07352efd7beb",
   "metadata": {},
   "outputs": [],
   "source": [
    "from typing import Type, Any, Callable, Union, List, Optional\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "from torch import Tensor\n",
    "\n",
    "# from model._internally_replaced_utils import load_state_dict_from_url\n",
    "\n",
    "\n",
    "__all__ = [\n",
    "    \"ResNet\",\n",
    "    \"resnet18\",\n",
    "    \"resnet34\",\n",
    "    \"resnet50\",\n",
    "    \"resnet101\",\n",
    "    \"resnet152\",\n",
    "    \"resnext50_32x4d\",\n",
    "    \"resnext101_32x8d\",\n",
    "    \"wide_resnet50_2\",\n",
    "    \"wide_resnet101_2\",\n",
    "]\n",
    "\n",
    "\n",
    "model_urls = {\n",
    "    \"resnet18\": \"https://download.pytorch.org/models/resnet18-f37072fd.pth\",\n",
    "    \"resnet34\": \"https://download.pytorch.org/models/resnet34-b627a593.pth\",\n",
    "    \"resnet50\": \"https://download.pytorch.org/models/resnet50-0676ba61.pth\",\n",
    "    \"resnet101\": \"https://download.pytorch.org/models/resnet101-63fe2227.pth\",\n",
    "    \"resnet152\": \"https://download.pytorch.org/models/resnet152-394f9c45.pth\",\n",
    "    \"resnext50_32x4d\": \"https://download.pytorch.org/models/resnext50_32x4d-7cdf4587.pth\",\n",
    "    \"resnext101_32x8d\": \"https://download.pytorch.org/models/resnext101_32x8d-8ba56ff5.pth\",\n",
    "    \"wide_resnet50_2\": \"https://download.pytorch.org/models/wide_resnet50_2-95faca4d.pth\",\n",
    "    \"wide_resnet101_2\": \"https://download.pytorch.org/models/wide_resnet101_2-32ee1156.pth\",\n",
    "}\n",
    "\n",
    "\n",
    "def conv3x3(in_planes: int, out_planes: int, stride: int = 1, groups: int = 1, dilation: int = 1) -> nn.Conv2d:\n",
    "    \"\"\"3x3 convolution with padding\"\"\"\n",
    "    return nn.Conv2d(\n",
    "        in_planes,\n",
    "        out_planes,\n",
    "        kernel_size=3,\n",
    "        stride=stride,\n",
    "        padding=dilation,\n",
    "        groups=groups,\n",
    "        bias=False,\n",
    "        dilation=dilation,\n",
    "    )\n",
    "\n",
    "\n",
    "def conv1x1(in_planes: int, out_planes: int, stride: int = 1) -> nn.Conv2d:\n",
    "    \"\"\"1x1 convolution\"\"\"\n",
    "    return nn.Conv2d(in_planes, out_planes, kernel_size=1, stride=stride, bias=False)\n",
    "\n",
    "\n",
    "class BasicBlock(nn.Module):\n",
    "    expansion: int = 1\n",
    "\n",
    "    def __init__(\n",
    "        self,\n",
    "        inplanes: int,\n",
    "        planes: int,\n",
    "        stride: int = 1,\n",
    "        downsample: Optional[nn.Module] = None,\n",
    "        groups: int = 1,\n",
    "        base_width: int = 64,\n",
    "        dilation: int = 1,\n",
    "        norm_layer: Optional[Callable[..., nn.Module]] = None,\n",
    "    ) -> None:\n",
    "        super(BasicBlock, self).__init__()\n",
    "        if norm_layer is None:\n",
    "            norm_layer = nn.BatchNorm2d\n",
    "        if groups != 1 or base_width != 64:\n",
    "            raise ValueError(\"BasicBlock only supports groups=1 and base_width=64\")\n",
    "        if dilation > 1:\n",
    "            raise NotImplementedError(\"Dilation > 1 not supported in BasicBlock\")\n",
    "        # Both self.conv1 and self.downsample layers downsample the input when stride != 1\n",
    "        self.conv1 = conv3x3(inplanes, planes, stride)\n",
    "        self.bn1 = norm_layer(planes)\n",
    "        self.relu = nn.ReLU(inplace=True)\n",
    "        self.conv2 = conv3x3(planes, planes)\n",
    "        self.bn2 = norm_layer(planes)\n",
    "        self.downsample = downsample\n",
    "        self.stride = stride\n",
    "\n",
    "    def forward(self, x: Tensor) -> Tensor:\n",
    "        identity = x\n",
    "\n",
    "        out = self.conv1(x)\n",
    "        out = self.bn1(out)\n",
    "        out = self.relu(out)\n",
    "\n",
    "        out = self.conv2(out)\n",
    "        out = self.bn2(out)\n",
    "\n",
    "        if self.downsample is not None:\n",
    "            identity = self.downsample(x)\n",
    "\n",
    "        out += identity\n",
    "        out = self.relu(out)\n",
    "\n",
    "        return out\n",
    "\n",
    "\n",
    "class Bottleneck(nn.Module):\n",
    "    # Bottleneck in torchvision places the stride for downsampling at 3x3 convolution(self.conv2)\n",
    "    # while original implementation places the stride at the first 1x1 convolution(self.conv1)\n",
    "    # according to \"Deep residual learning for image recognition\"https://arxiv.org/abs/1512.03385.\n",
    "    # This variant is also known as ResNet V1.5 and improves accuracy according to\n",
    "    # https://ngc.nvidia.com/catalog/model-scripts/nvidia:resnet_50_v1_5_for_pytorch.\n",
    "\n",
    "    expansion: int = 4\n",
    "\n",
    "    def __init__(\n",
    "        self,\n",
    "        inplanes: int,\n",
    "        planes: int,\n",
    "        stride: int = 1,\n",
    "        downsample: Optional[nn.Module] = None,\n",
    "        groups: int = 1,\n",
    "        base_width: int = 64,\n",
    "        dilation: int = 1,\n",
    "        norm_layer: Optional[Callable[..., nn.Module]] = None,\n",
    "    ) -> None:\n",
    "        super(Bottleneck, self).__init__()\n",
    "        if norm_layer is None:\n",
    "            norm_layer = nn.BatchNorm2d\n",
    "        width = int(planes * (base_width / 64.0)) * groups\n",
    "        # Both self.conv2 and self.downsample layers downsample the input when stride != 1\n",
    "        self.conv1 = conv1x1(inplanes, width)\n",
    "        self.bn1 = norm_layer(width)\n",
    "        self.conv2 = conv3x3(width, width, stride, groups, dilation)\n",
    "        self.bn2 = norm_layer(width)\n",
    "        self.conv3 = conv1x1(width, planes * self.expansion)\n",
    "        self.bn3 = norm_layer(planes * self.expansion)\n",
    "        self.relu = nn.ReLU(inplace=True)\n",
    "        self.downsample = downsample\n",
    "        self.stride = stride\n",
    "\n",
    "    def forward(self, x: Tensor) -> Tensor:\n",
    "        identity = x\n",
    "\n",
    "        out = self.conv1(x)\n",
    "        out = self.bn1(out)\n",
    "        out = self.relu(out)\n",
    "\n",
    "        out = self.conv2(out)\n",
    "        out = self.bn2(out)\n",
    "        out = self.relu(out)\n",
    "\n",
    "        out = self.conv3(out)\n",
    "        out = self.bn3(out)\n",
    "\n",
    "        if self.downsample is not None:\n",
    "            identity = self.downsample(x)\n",
    "\n",
    "        out += identity\n",
    "        out = self.relu(out)\n",
    "\n",
    "        return out\n",
    "\n",
    "\n",
    "class ResNet(nn.Module):\n",
    "    def __init__(\n",
    "        self,\n",
    "        block: Type[Union[BasicBlock, Bottleneck]],\n",
    "        layers: List[int],\n",
    "        num_classes: int = 1000,\n",
    "        zero_init_residual: bool = False,\n",
    "        groups: int = 1,\n",
    "        width_per_group: int = 64,\n",
    "        replace_stride_with_dilation: Optional[List[bool]] = None,\n",
    "        norm_layer: Optional[Callable[..., nn.Module]] = None,\n",
    "    ) -> None:\n",
    "        super(ResNet, self).__init__()\n",
    "        if norm_layer is None:\n",
    "            norm_layer = nn.BatchNorm2d\n",
    "        self._norm_layer = norm_layer\n",
    "\n",
    "        self.inplanes = 64\n",
    "        self.dilation = 1\n",
    "        if replace_stride_with_dilation is None:\n",
    "            # each element in the tuple indicates if we should replace\n",
    "            # the 2x2 stride with a dilated convolution instead\n",
    "            replace_stride_with_dilation = [False, False, False]\n",
    "        if len(replace_stride_with_dilation) != 3:\n",
    "            raise ValueError(\n",
    "                \"replace_stride_with_dilation should be None \"\n",
    "                \"or a 3-element tuple, got {}\".format(replace_stride_with_dilation)\n",
    "            )\n",
    "        self.groups = groups\n",
    "        self.base_width = width_per_group\n",
    "        self.conv1 = nn.Conv2d(3, self.inplanes, kernel_size=7, stride=2, padding=3, bias=False)\n",
    "        self.bn1 = norm_layer(self.inplanes)\n",
    "        self.relu = nn.ReLU(inplace=True)\n",
    "        self.maxpool = nn.MaxPool2d(kernel_size=3, stride=2, padding=1)\n",
    "        self.layer1 = self._make_layer(block, 64, layers[0])\n",
    "        self.layer2 = self._make_layer(block, 128, layers[1], stride=2, dilate=replace_stride_with_dilation[0])\n",
    "        self.layer3 = self._make_layer(block, 256, layers[2], stride=2, dilate=replace_stride_with_dilation[1])\n",
    "        self.layer4 = self._make_layer(block, 512, layers[3], stride=2, dilate=replace_stride_with_dilation[2])\n",
    "        self.avgpool = nn.AdaptiveAvgPool2d((1, 1))\n",
    "        self.fc = nn.Linear(512 * block.expansion, num_classes)\n",
    "\n",
    "        for m in self.modules():\n",
    "            if isinstance(m, nn.Conv2d):\n",
    "                nn.init.kaiming_normal_(m.weight, mode=\"fan_out\", nonlinearity=\"relu\")\n",
    "            elif isinstance(m, (nn.BatchNorm2d, nn.GroupNorm)):\n",
    "                nn.init.constant_(m.weight, 1)\n",
    "                nn.init.constant_(m.bias, 0)\n",
    "\n",
    "        # Zero-initialize the last BN in each residual branch,\n",
    "        # so that the residual branch starts with zeros, and each residual block behaves like an identity.\n",
    "        # This improves the model by 0.2~0.3% according to https://arxiv.org/abs/1706.02677\n",
    "        if zero_init_residual:\n",
    "            for m in self.modules():\n",
    "                if isinstance(m, Bottleneck):\n",
    "                    nn.init.constant_(m.bn3.weight, 0)  # type: ignore[arg-type]\n",
    "                elif isinstance(m, BasicBlock):\n",
    "                    nn.init.constant_(m.bn2.weight, 0)  # type: ignore[arg-type]\n",
    "\n",
    "    def _make_layer(\n",
    "        self,\n",
    "        block: Type[Union[BasicBlock, Bottleneck]],\n",
    "        planes: int,\n",
    "        blocks: int,\n",
    "        stride: int = 1,\n",
    "        dilate: bool = False,\n",
    "    ) -> nn.Sequential:\n",
    "        norm_layer = self._norm_layer\n",
    "        downsample = None\n",
    "        previous_dilation = self.dilation\n",
    "        if dilate:\n",
    "            self.dilation *= stride\n",
    "            stride = 1\n",
    "        if stride != 1 or self.inplanes != planes * block.expansion:\n",
    "            downsample = nn.Sequential(\n",
    "                conv1x1(self.inplanes, planes * block.expansion, stride),\n",
    "                norm_layer(planes * block.expansion),\n",
    "            )\n",
    "\n",
    "        layers = []\n",
    "        layers.append(\n",
    "            block(\n",
    "                self.inplanes, planes, stride, downsample, self.groups, self.base_width, previous_dilation, norm_layer\n",
    "            )\n",
    "        )\n",
    "        self.inplanes = planes * block.expansion\n",
    "        for _ in range(1, blocks):\n",
    "            layers.append(\n",
    "                block(\n",
    "                    self.inplanes,\n",
    "                    planes,\n",
    "                    groups=self.groups,\n",
    "                    base_width=self.base_width,\n",
    "                    dilation=self.dilation,\n",
    "                    norm_layer=norm_layer,\n",
    "                )\n",
    "            )\n",
    "\n",
    "        return nn.Sequential(*layers)\n",
    "        \n",
    "    def init_weights(self, *args, **kwargs) -> None:\n",
    "        # def _init_weights(m: nn.Module):\n",
    "        #     if isinstance(m, nn.Linear) or isinstance(m, nn.Conv2d):\n",
    "        #         nn.init.xavier_uniform_(m.weight)\n",
    "        #         nn.init.zeros_(m.bias)\n",
    "        #self.apply(_init_weights)\n",
    "        pass\n",
    "\n",
    "    def _forward_impl(self, x: Tensor) -> Tensor:\n",
    "        # See note [TorchScript super()]\n",
    "        x = self.conv1(x)\n",
    "        x = self.bn1(x)\n",
    "        x = self.relu(x)\n",
    "        x = self.maxpool(x)\n",
    "\n",
    "        x = self.layer1(x)\n",
    "        x = self.layer2(x)\n",
    "        x = self.layer3(x)\n",
    "        x = self.layer4(x)\n",
    "\n",
    "        x = self.avgpool(x)\n",
    "        feat = torch.flatten(x, 1)\n",
    "        x = self.fc(feat)\n",
    "\n",
    "        return feat, x\n",
    "\n",
    "    def forward(self, x: Tensor, latent_output: bool = False) -> Tensor:\n",
    "        feat, out = self._forward_impl(x)\n",
    "        if  latent_output == False:\n",
    "            output = out\n",
    "        else:\n",
    "            output = feat\n",
    "        return output\n",
    "\n",
    "\n",
    "def _resnet(\n",
    "    arch: str,\n",
    "    block: Type[Union[BasicBlock, Bottleneck]],\n",
    "    layers: List[int],\n",
    "    pretrained: bool,\n",
    "    progress: bool,\n",
    "    **kwargs: Any,\n",
    ") -> ResNet:\n",
    "    model = ResNet(block, layers, **kwargs)\n",
    "    # if pretrained:\n",
    "    #     state_dict = load_state_dict_from_url(model_urls[arch], progress=progress)\n",
    "    #     model.load_state_dict(state_dict)\n",
    "    return model\n",
    "\n",
    "\n",
    "def ResNet50(pretrained: bool = False, progress: bool = True, **kwargs: Any) -> ResNet:\n",
    "    r\"\"\"ResNet-50 model from\n",
    "    `\"Deep Residual Learning for Image Recognition\" <https://arxiv.org/pdf/1512.03385.pdf>`_.\n",
    "    Args:\n",
    "        pretrained (bool): If True, returns a model pre-trained on ImageNet\n",
    "        progress (bool): If True, displays a progress bar of the download to stderr\n",
    "    \"\"\"\n",
    "    return _resnet(\"resnet50\", Bottleneck, [3, 4, 6, 3], pretrained, progress, **kwargs)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "9e08faa4-e0d0-4358-b1c7-fb5a17410fb2",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "\n",
    "net = ResNet50(pretrained=False)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "a9854273-eecc-46c7-a991-e3aae1db832f",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/guan/miniconda3/envs/tf/lib/python3.10/site-packages/torchvision/models/_utils.py:208: UserWarning: The parameter 'pretrained' is deprecated since 0.13 and may be removed in the future, please use 'weights' instead.\n",
      "  warnings.warn(\n",
      "/home/guan/miniconda3/envs/tf/lib/python3.10/site-packages/torchvision/models/_utils.py:223: UserWarning: Arguments other than a weight enum or `None` for 'weights' are deprecated since 0.13 and may be removed in the future. The current behavior is equivalent to passing `weights=ResNet50_Weights.IMAGENET1K_V1`. You can also use `weights=ResNet50_Weights.DEFAULT` to get the most up-to-date weights.\n",
      "  warnings.warn(msg)\n"
     ]
    }
   ],
   "source": [
    "model = models.resnet50(pretrained=True)\n",
    "net.load_state_dict(model.state_dict())\n",
    "net.fc = nn.Linear(2048, args.num_classes)\n",
    "# netglob = netglob.to(args.device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "d2b6f875-eee9-48f0-925a-bf760b6e0614",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([32, 14])\n"
     ]
    }
   ],
   "source": [
    "dummy_input = torch.randn(32,3, 32, 32)  # Batch size of 32\n",
    "output = net(dummy_input)\n",
    "print(output.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "3d1ec6f5-098d-4490-b84b-0a6ea2673f8a",
   "metadata": {},
   "outputs": [],
   "source": [
    "batch_size = 16"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "cf00ce1c-9b64-48e6-a5ed-8fea88ca3790",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<__main__.Clothing at 0x700710672440>"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "train_dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "5f6741d4-5d4a-4dfe-829d-d3ac20d5e18a",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_loader = DataLoader(train_dataset, batch_size=batch_size, shuffle=True, num_workers=2)\n",
    "test_loader = DataLoader(test_dataset, batch_size=batch_size, shuffle=False, num_workers=2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "35bd6b39-8ef2-46bc-93a7-9d982b19b3d6",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_data1, train_targets1 = [], []\n",
    "for batch in train_loader:\n",
    "    images, labels = batch\n",
    "    train_data1.append(images)\n",
    "    train_targets1.append(labels)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "56030961-8e47-481c-a15b-4d1eb54256d3",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_data = torch.cat(train_data1, dim=0)\n",
    "train_targets = torch.cat(train_targets1, dim=0)\n",
    "\n",
    "test_data1, test_targets1 = [], []\n",
    "for batch in test_loader:\n",
    "    images, labels = batch\n",
    "    test_data1.append(images)\n",
    "    test_targets1.append(labels)\n",
    "\n",
    "test_data = torch.cat(test_data1, dim=0)\n",
    "test_targets = torch.cat(test_targets1, dim=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ec544b6d-bff3-428f-98d6-a34bd2218a8e",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "c9097031-2699-44ab-8d4c-5f4fb39595ca",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(train_data, 'train_data.pt')\n",
    "torch.save(train_targets, 'train_targets.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "1eae6bdd-25f9-4b10-8b21-808a091b5151",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(test_data, 'test_data.pt')\n",
    "torch.save(test_targets, 'test_targets.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6395c14e-0bf0-4dc2-afb6-79abd6b6c6cc",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d631dd54-c093-406a-8414-e94da906fdb6",
   "metadata": {},
   "outputs": [],
   "source": [
    "Xtr, ytr = (train_data, train_targets)\n",
    "Xte, yte = (test_data, test_targets)\n",
    "\n",
    "print(\"this is\")\n",
    "print(ytr[1])\n",
    "data_handler = ClassificationDataHandler(Xtr, ytr,\n",
    "                                         Xte, yte)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "f95ab78a-a13e-411a-b3f1-49eef0298814",
   "metadata": {},
   "outputs": [],
   "source": [
    "data_dispatcher = CustomDataDispatcher(data_handler, n=10, eval_on_user=False, auto_assign=True)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "abdae7b4-a5b1-4c3a-9712-061e24c91046",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([10,  8,  9,  ...,  0, 11, 11])"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "data_dispatcher[3][0][1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "b0c81d8c-7045-4c97-ac15-74328bfcaf02",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjsAAAHHCAYAAABZbpmkAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy81sbWrAAAACXBIWXMAAA9hAAAPYQGoP6dpAAA80klEQVR4nO3deVhWdf7/8Rc7yqooWyrgkkq5lCtlZUqQoeZoi45jqNmiYC4zZlYqao7LTGYamZZhmy02LWamIC5NihtGuaRpuU0KZAqIyiKc3x/9uL/egRvCfePp+biuc+U553Of9/vc0u2Ls90OhmEYAgAAMClHezcAAABQnQg7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7wJ/Y4MGDFRoaapNaoaGhGjx4sGV+yZIlcnBw0Pbt221Sv2vXruratatNav3R/v37FRUVJR8fHzk4OOizzz6zSx8XcnBwUEJCgr3bAGyCsANUoOwf4rLJ3d1dwcHBio6O1rx583T69OlKb3vTpk1KSEhQTk5O1TUsKSEhwarn2rVrq1GjRurVq5eSkpJUWFhYJXX27NmjhIQEHTp0qEq2V5Vqam+xsbHauXOnpk+frnfeeUft27e3d0s10tdff63evXurYcOGcnd3V2BgoO69915t3LjR3q3hOuds7waAmmzq1KkKCwtTcXGxMjMztX79eo0ePVpz5szR8uXL1bp166ve5qZNmzRlyhQNHjxYvr6+Vd7zggUL5OnpqcLCQv3yyy9avXq1hg4dqrlz52rFihVq2LChZezrr7+u0tLSq9r+nj17NGXKFHXt2vWqjgrt27dPjo7V+/vVpXpLTk6u1toXc+7cOaWlpem5555TfHy8XXq4Xvz4449ydHTUk08+qcDAQJ06dUrvvvuu7rzzTn355Ze699577d0irlOEHeASevToYfVb+IQJE7R27Vr17NlTvXv31g8//KBatWrZscPyHnjgAdWrV88yP2nSJL333nt65JFH9OCDD2rz5s2WdS4uLtXai2EYKigoUK1ateTm5lattS7H1dXVLnV//fVXSarSYHvmzBl5eHhU2fZqimHDhmnYsGFWy0aMGKHGjRtr7ty5hB1UGqexgKvUrVs3TZw4UYcPH9a7775rWf79999r8ODBaty4seUQ/NChQ/Xbb79ZxiQkJGjcuHGSpLCwMMspp7LTLklJSerWrZv8/f3l5uam8PBwLViw4Jp7HjhwoIYNG6YtW7YoJSXFsryia3Y++OADtWvXTl5eXvL29larVq308ssvS/r99N6DDz4oSbr77rst/a9fv17S79fl9OzZU6tXr1b79u1Vq1YtLVy40LLuwmt2ypw9e1ZPPPGE/Pz85O3trUceeUSnTp2yGnOx60su3Obleqvomp3s7Gw9+uijCggIkLu7u9q0aaO33nrLasyhQ4fk4OCgf//731q0aJGaNGkiNzc3dejQQdu2bavw/S6TkJCgkJAQSdK4cePk4OBg9X5/++236tGjh7y9veXp6anu3btbhdGy/XJwcNCGDRs0YsQI+fv7q0GDBpesW1BQoISEBN14441yd3dXUFCQ+vbtq59++umirzl8+LBGjBih5s2bq1atWvLz89ODDz5Y7pRgcXGxpkyZombNmsnd3V1+fn7q0qWL1c9VZmamhgwZogYNGsjNzU1BQUG6//77K3V6sXbt2qpfv36Vn/bFnwtHdoBKGDRokJ599lklJyfrsccekySlpKTo559/1pAhQxQYGKjdu3dr0aJF2r17tzZv3iwHBwf17dtXP/74o95//3299NJLliMw9evXl/T7KaibbrpJvXv3lrOzs7744guNGDFCpaWliouLu+aeFy1apOTkZN1zzz0VjklJSdGAAQPUvXt3zZo1S5L0ww8/aOPGjRo1apTuvPNOPfXUU5o3b56effZZtWzZUpIs/5V+P101YMAAPfHEE3rsscfUvHnzS/YVHx8vX19fJSQkaN++fVqwYIEOHz6s9evXy8HB4Yr370p6u9C5c+fUtWtXHThwQPHx8QoLC9OyZcs0ePBg5eTkaNSoUVbjly5dqtOnT+uJJ56Qg4ODZs+erb59++rnn3++6BGyvn37ytfXV2PGjNGAAQN03333ydPTU5K0e/du3XHHHfL29tbTTz8tFxcXLVy4UF27dtWGDRvUqVMnq22NGDFC9evX16RJk3TmzJmLvg8lJSXq2bOnUlNT1b9/f40aNUqnT59WSkqKdu3apSZNmlT4um3btmnTpk3q37+/GjRooEOHDmnBggXq2rWr9uzZo9q1a0v6PcDNmDFDw4YNU8eOHZWXl6ft27drx44dlp+rfv36affu3Ro5cqRCQ0OVnZ2tlJQUHTly5IpOfebl5amoqEgnTpzQ22+/rV27dunZZ5+97OuAizIAlJOUlGRIMrZt23bRMT4+PsYtt9ximT979my5Me+//74hyfj6668ty/71r38ZkoyDBw+WG1/RNqKjo43GjRtftufJkycbkoxff/21wvWnTp0yJBl/+ctfLMtiY2ONkJAQy/yoUaMMb29v4/z58xets2zZMkOSsW7dunLrQkJCDEnGqlWrKlwXGxtrmS97j9u1a2cUFRVZls+ePduQZHz++eeWZZKMyZMnX3abl+rtrrvuMu666y7L/Ny5cw1JxrvvvmtZVlRUZERERBienp5GXl6eYRiGcfDgQUOS4efnZ5w8edIy9vPPPzckGV988UW5Whcqe/2//vUvq+V9+vQxXF1djZ9++smy7NixY4aXl5dx5513WpaVvU9dunS55N9LmTfffNOQZMyZM6fcutLSUsuf//ieVvSzl5aWZkgy3n77bcuyNm3aGDExMRetX/Zz9sf9vRrR0dGGJEOS4erqajzxxBPGuXPnKr09gNNYQCV5enpa3ZV14bU7BQUFOnHihDp37ixJ2rFjxxVt88Jt5Obm6sSJE7rrrrv0888/Kzc395r7lXTJO8l8fX115swZq1MSVyssLEzR0dFXPP7xxx+3OjIyfPhwOTs7a+XKlZXu4UqsXLlSgYGBGjBggGWZi4uLnnrqKeXn52vDhg1W4x9++GHVqVPHMn/HHXdIkn7++eerrl1SUqLk5GT16dNHjRs3tiwPCgrSX//6V33zzTfKy8uzes1jjz0mJyeny277P//5j+rVq6eRI0eWW3epI2UX/uwVFxfrt99+U9OmTeXr62v18+vr66vdu3dr//79F92Oq6ur1q9fX+505JWaOXOmkpOTtXjxYnXu3FlFRUU6f/58pbYFSFyzA1Rafn6+vLy8LPMnT57UqFGjFBAQoFq1aql+/foKCwuTpCsOKhs3blRkZKQ8PDzk6+ur+vXrWw7fX2vYyc/PlySrnv9oxIgRuvHGG9WjRw81aNBAQ4cO1apVq66qTtk+X6lmzZpZzXt6eiooKKjabx8/fPiwmjVrVu4OsbLTXocPH7Za3qhRI6v5suBTmX/Qf/31V509e7bCU3wtW7ZUaWmpjh49arX8St/Xn376Sc2bN5ez89VdpXDu3DlNmjRJDRs2lJubm+rVq2e5VubCn72pU6cqJydHN954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xhtv1IEDBxQYGKiioiLl5ORYjcnKyrJc4xMYGFju7qyy+YquAyrj5uYmb29vqwkAAJhTjQo7+fn5+umnnxQUFKR27drJxcVFqamplvX79u3TkSNHFBERIUmKiIjQzp07lZ2dbRmTkpIib29vhYeH27x/AABQ89j1NNY//vEP9erVSyEhITp27JgmT54sJycnDRgwQD4+Pnr00Uc1duxY1a1bV97e3ho5cqQiIiLUuXNnSVJUVJTCw8M1aNAgzZ49W5mZmXr++ecVFxcnNzc3e+4aAACoIewadv73v/9pwIAB+u2331S/fn116dJFmzdvVv369SVJL730khwdHdWvXz8VFhYqOjpar776quX1Tk5OWrFihYYPH66IiAh5eHgoNjZWU6dOtdcuAbgEHkEAwB7sGnY++OCDS653d3dXYmKiEhMTLzomJCREK1eurOrWAACASdj9oYIAbIcjKwD+jGrUBcoAAABVjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMjbADAABMzdneDQAAAPsIfeZLm9Q5NDPGJnUuhiM7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Ag7AADA1Jzt3QAA/NmEPvOlTeocmhljkzpATceRHQAAYGqEHQAAYGqEHQAAYGqEHQAAYGqEHQAAYGqEHQAAYGqEHQAAYGqEHQAAYGqEHQAAYGqEHQAAYGqEHQAAYGqEHQAAYGo1JuzMnDlTDg4OGj16tGVZQUGB4uLi5OfnJ09PT/Xr109ZWVlWrzty5IhiYmJUu3Zt+fv7a9y4cTp//ryNuwcAADVVjQg727Zt08KFC9W6dWur5WPGjNEXX3yhZcuWacOGDTp27Jj69u1rWV9SUqKYmBgVFRVp06ZNeuutt7RkyRJNmjTJ1rsAAABqKLuHnfz8fA0cOFCvv/666tSpY1mem5urxYsXa86cOerWrZvatWunpKQkbdq0SZs3b5YkJScna8+ePXr33XfVtm1b9ejR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hs17kWR8+umnNq97oezsbEOSsWHDBrv2UadOHeONN96wac3Tp08bzZo1M1JSUoy77rrLGDVqlM1qT5482WjTpo3N6lVk/PjxRpcuXezawx+NGjXKaNKkiVFaWmqTejExMcbQoUOtlvXt29cYOHCgTeobhmGcPXvWcHJyMlasWGG1/NZbbzWee+65aq//x8+h0tJSIzAw0PjXv/5lWZaTk2O4ubkZ77//frXXv9DBgwcNSca3335b5XWvtIcyW7duNSQZhw8ftlsPubm5hiRjzZo11dLDteLITg1RVFSk9PR0RUZGWpY5OjoqMjJSaWlpduzMfnJzcyVJdevWtUv9kpISffDBBzpz5owiIiJsWjsuLk4xMTFWPw+2tH//fgUHB6tx48YaOHCgjhw5YtP6y5cvV/v27fXggw/K399ft9xyi15//XWb9nChoqIivfvuuxo6dKjNviz4tttuU2pqqn788UdJ0nfffadvvvlGPXr0sEl9STp//rxKSkrk7u5utbxWrVo2P9onSQcPHlRmZqbV/xc+Pj7q1KnTn/ZzUvr9s9LBwcFu3/FYVFSkRYsWycfHR23atLFLD5djmicoX+9OnDihkpKScl9xERAQoL1799qpK/spLS3V6NGjdfvtt+vmm2+2ae2dO3cqIiJCBQUF8vT01Keffqrw8HCb1f/ggw+0Y8eOar0u4lI6deqkJUuWqHnz5jp+/LimTJmiO+64Q7t27ZKXl5dNevj555+1YMECjR07Vs8++6y2bdump556Sq6uroqNjbVJDxf67LPPlJOTo8GDB9us5jPPPKO8vDy1aNFCTk5OKikp0fTp0zVw4ECb9eDl5aWIiAhNmzZNLVu2VEBAgN5//32lpaWpadOmNuujTGZmpiRV+DlZtu7PpqCgQOPHj9eAAQNs/sWcK1asUP/+/XX27FkFBQUpJSVF9erVs2kPV4qwgxopLi5Ou3btsstvj82bN1dGRoZyc3P18ccfKzY2Vhs2bLBJ4Dl69KhGjRqllJSUcr9N28qFRw5at26tTp06KSQkRB999JEeffRRm/RQWlqq9u3b65///Kck6ZZbbtGuXbv02muv2SXsLF68WD169FBwcLDNan700Ud67733tHTpUt10003KyMjQ6NGjFRwcbNP34J133tHQoUN1ww03yMnJSbfeeqsGDBig9PR0m/WAihUXF+uh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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "# Python\n",
    "\n",
    "import torch\n",
    "import matplotlib.pyplot as plt\n",
    "from collections import Counter\n",
    "\n",
    "data = data_dispatcher[2][0][1]\n",
    "\n",
    "# Count the occurrences of each label\n",
    "label_distribution = Counter(data.numpy())\n",
    "\n",
    "# Plot the data distribution\n",
    "labels, counts = zip(*label_distribution.items())\n",
    "plt.bar(labels, counts, tick_label=labels)\n",
    "plt.xlabel('Labels')\n",
    "plt.ylabel('Count')\n",
    "plt.title('Data Distribution for class 3')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "6ed58acd-9cf8-4cba-8c3a-96da59148cfc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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oVbz5cHnl/cXAWbGPTvAHeT7779Xfn6+mjZtalnWv39/bd++XePHj1e7du3k4+OjsrIyPfzwwzf0ePd///tf9ejRQy1atNCcOXPUsGFDubu7a82aNZo7d+5NPyL+S/v375ckq55/KSAgQBkZGVq/fr3Wrl2rtWvXKjk5WUOGDKlw4+712PLptNLSUpvVcnFxueZywzBs1gNgb4Qd4A7y/vvvS5Llcs358+eVmpqqadOmafLkyZZxR44cqbDt9W66XbVqlYqKirRy5UqrswibN2+ulp6vx93dXb1791bv3r1VVlam559/Xm+//bYmTZqkpk2b3tRNwzfiyJEjevDBBy3zhYWFOnv2rB555BHLslq1alV4CWNxcbHOnj1rtexmegsLC9M333yjsrIyq7M7hw8ftqwHYI3LWMAdYtOmTZoxY4bCw8MVGxsr6f9+6//lb/nz5s2rsL23t7ckVfjH+1r7yM/PV3Jy8i33vGzZMv39739XZGSkevTocd1xVz8mL0nOzs6WFyeWv4H5ev1X1uLFi1VSUmKZX7hwoa5cuaJevXpZljVp0kRfffVVhe1+eWbnZnp75JFHlJWVpY8//tiy7MqVK1qwYIF8fHz0wAMPVOZwAFPjzA5gQmvXrtXhw4d15coVZWdna9OmTUpJSVFYWJhWrlwpT09PSZKvr69+//vfa/bs2SopKVH9+vW1YcMGHTt2rMI+O3ToIEl66aWXNHDgQLm5ual3797q2bOn5azKc889p8LCQr3zzjsKCAiocAbj13z66afy8fFRcXGx5Q3K27ZtU9u2bbVixYpf3faZZ55Rbm6uunfvrgYNGujEiRNasGCB2rVrZ7mXpV27dnJxcdGsWbOUn58vDw8Py7uBKqO4uFg9evRQ//79lZmZqbfeekv333+/HnvsMau+/vjHP6pfv3566KGH9PXXX2v9+vVWL0+82d6effZZvf322xo6dKjS09PVqFEjffrpp9q2bZvmzZv3q/c2AXcsOz8NBqAKlT8WXT65u7sbQUFBxkMPPWS8+eablseSr/b9998bTzzxhOHv72/4+fkZTz31lHHmzJlrPjY9Y8YMo379+oazs7PVY+grV6402rRpY3h6ehqNGjUyZs2aZbz77rvXfVT9auWPnpdPnp6eRoMGDYxHH33UePfdd43Lly9X2OaXj55/+umnRs+ePY2AgADD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y9ddz8yZM+Xn52eZGjZsWLUHBgAAHIZdL2P16tXL8uc2bdqoS5cuCgsL0yeffCIvL69qqztx4kQlJiZa5gsKCgg8AACYlN0vY13N399fd911l44ePaqgoCAVFxcrLy/Pakx2drblHp+goKAKT2eVz1/rPqByHh4e8vX1tZoAAIA5OVTYKSws1H//+18FBwerQ4cOcnNzU2pqqmV9ZmamTp48qcjISElSZGSk9u3bp5ycHMuYlJQU+fr6KiIiwub9AwAAx2PXy1h/+tOf1Lt3b4WFhenMmTOaMmWKXFxcNGjQIPn5+Wn48OFKTExU7dq15evrq1GjRikyMlJdu3aVJPXs2VMREREaPHiwZs+eraysLL388suKj4+Xh4eHPQ8NAAA4CLuGne+//16DBg3Sjz/+qHr16un+++/Xjh07VK9ePUnS3Llz5ezsrH79+qmoqEjR0dF66623LNu7uLho9erVGjlypCIjI+Xt7a24uDhNnz7dXocEAAAcjF3DzvLly391vaenp5KSkpSUlHTdMWFhYVqzZk1VtwYAAEzCoe7ZAQAAqGqEHQAAYGqEHQAAYGp2/24sALbDd7UBuBNxZgcAAJgaZ3YA3FE4uwXceTizAwAATI2wAwAATI2wAwAATI2wAwAATI2wAwAATI2wAwAATI2wAwAATI337AAAcIe6U947xZkdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgaoQdAABgag4Tdl5//XU5OTlp7NixlmWXL19WfHy86tSpIx8fH/Xr10/Z2dlW2508eVIxMTGqUaOGAgICNH78eF25csXG3QMAAEflEGFn9+7devvtt9WmTRur5ePGjdOqVau0YsUKbd26VWfOnFHfvn0t60tLSxUTE6Pi4mJt375d7733npYuXarJkyfb+hAAAICDsnvYKSwsVGxsrN555x3VqlXLsjw/P19LlizR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qTcoS1KtWrXUqVOnquwFAACgylU67AC3k0YvfGmTOsdfj7FJHQDAjavUF4ECAADcLgg7AADA1Ag7AADA1Ag7AADA1LhBGQBgczw0AFvizA4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1wg4AADA1u4admTNnqlOnTqpZs6YCAgLUp08fZWZmWo25fPmy4uPjVadOHfn4+Khfv37Kzs62GnPy5EnFxMSoRo0aCggI0Pjx43XlyhVbHgoAAHBQdg07W7duVXx8vHbs2KGUlBSVlJSoZ8+eunjxomXMuHHjtGrVKq1YsUJbt27VmTNn1LdvX8v60tJSxcTEqLi4WNu3b9d7772npUuXavLkyfY4JAAA4GBc7Vl83bp1VvNLly5VQECA0tPT9fvf/175+flasmSJli1bpu7du0uSkpOT1bJlS+3YsUNdu3bVhg0bdPDgQW3cuFGBgYFq166dZsyYoQkTJmjq1Klyd3e3x6EBAAAH4VD37OTn50uSateuLUlKT09XSUmJoqKiLGNatGih0NBQpaWlSZLS0tLUunVrBQYGWsZER0eroKBABw4csGH3AADAEdn1zM7VysrKNHbsWN133326++67JUlZWVlyd3eXv7+/1djAwEBlZWVZxlwddMrXl6+7lqKiIhUVFVnmCwoKquowAACAg3GYMzvx8fHav3+/li9fXu21Zs6cKT8/P8vUsGHDaq8JAADswyHCTkJCglavXq3NmzerQYMGluVBQUEqLi5WXl6e1fjs7GwFBQVZxvzy6azy+fIxvzRx4kTl5+dbplOnTlXh0QAAAEdi17BjGIYSEhL02WefadOmTQoPD7da36FDB7m5uSk1NdWyLDMzUydPnlRkZKQkKTIyUvv27VNOTo5lTEpKinx9fRUREXHNuh4eHvL19bWaAACAOdn1np34+HgtW7ZMX3zxhWrWrGm5x8bPz09eXl7y8/PT8OHDlZiYqNq1a8vX11ejRo1SZGSkunbtKknq2bOnIiIiNHjwYM2ePVtZWVl6+eWXFR8fLw8PD3seHgAAcAB2DTsLFy6UJHXr1s1qeXJysoYOHSpJmjt3rpydndWvXz8VFRUpOjpab731lmWsi4uLVq9erZEjRyoyMlLe3t6Ki4vT9OnTbXUYAADAgdk17BiG8ZtjPD09lZSUpKSkpOuOCQsL05o1a6qyNQAAYBIOcYMyAABAdSHsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAUyPsAAAAU3O1dwPAnaLRC1/apM7x12NsUgcAbhd2PbPz1VdfqXfv3goJCZGTk5M+//xzq/WGYWjy5MkKDg6Wl5eXoqKidOTIEasxubm5io2Nla+vr/z9/TV8+HAVFhba8CgAAIAjs2vYuXjxotq2baukpKRrrp89e7bmz5+vRYsWaefOnfL29lZ0dLQuX75sGRMbG6sDBw4oJSVFq1ev1ldffaVnn33WVocAAAAcnF0vY/Xq1Uu9evW65jrDMDRv3jy9/PLLevzxxyVJ//jHPxQYGKjPP/9cAwcO1KFDh7Ru3Trt3r1bHTt2lCQtWLBAjzzyiP76178qJCTEZscCAAAck8PeoHzs2DFlZWUpKirKsszPz09dunRRWlqaJCktLU3+/v6WoCNJUVFRcnZ21s6dO6+776KiIhUUFFhNAADAnBw27GRlZUmSAgMDrZYHBgZa1mVlZSkgIMBqvaurq2rXrm0Zcy0zZ86Un5+fZWrYsGEVdw8AAByFw4ad6jRx4kTl5+dbplOnTtm7JQAAUE0cNuwEBQVJkrKzs62WZ2dnW9YFBQUpJyfHav2VK1eUm5trGXMtHh4e8vX1tZoAAIA5OWzYCQ8PV1BQkFJTUy3LCgoKtHPnTkVGRkqSIiMjlZeXp/T0dMuYTZs2qaysTF26dLF5zwAAwPHY9WmswsJCHT161DJ/7NgxZWRkqHbt2goNDdXYsWP1yiuvqFmzZgoPD9ekSZMUEhKiPn36SJJatmyphx9+WCNGjNCiRYtUUlKihIQEDRw4kCexAACAJDuHnT179ujBBx+0zCcmJkqS4uLitHTpUv35z3/WxYsX9eyzzyovL0/333+/1q1bJ09PT8s2H374oRISEtSjRw85OzurX79+mj9/vs2PBQAAOCa7hp1u3brJMIzrrndyctL06dM1ffr0646pXbu2li1bVh3tAQAAE3DYe3YAAACqAmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYGmEHAACYmmnCTlJSkho1aiRPT0916dJFu3btsndLAADAAZgi7Hz88cdKTEzUlClTtHfvXrVt21bR0dHKycmxd2sAAMDOTBF25syZoxEjRujpp59WRESEFi1apBo1aujdd9+1d2sAAMDObvuwU1xcrPT0dEVFRVmWOTs7KyoqSmlpaXbsDAAAOAJXezdwq86dO6fS0lIFBgZaLQ8MDNThw4evuU1RUZGKioos8/n5+ZKkgoKCKu+vrOhSle/zWn6td3qwf316cIz69OAY9enBMeo7Sg9VsV/DMH59oHGbO336tCHJ2L59u9Xy8ePHG507d77mNlOmTDEkMTExMTExMZlgOnXq1K9mhdv+zE7dunXl4uKi7Oxsq+XZ2dkKCgq65jYTJ05UYmKiZb6srEy5ubmqU6eOnJycqrXf31JQUKCGDRvq1KlT8vX1pQc79WDv+vTgGPXpwTHq04Nj1HeUHq5mGIYuXLigkJCQXx1324cdd3d3dejQQampqerTp4+kn8NLamqqEhISrrmNh4eHPDw8rJb5+/tXc6c3x9fX1+7/IdGD/evTg2PUpwfHqE8PjlHfUXoo5+fn95tjbvuwI0mJiYmKi4tTx44d1blzZ82bN08XL17U008/be/WAACAnZki7AwYMEA//PCDJk+erKysLLVr107r1q2rcNMyAAC485gi7EhSQkLCdS9b3U48PDw0ZcqUCpfZ6OHOqk8PjlGfHhyjPj04Rn1H6aEynAzjt57XAgAAuH3d9i8VBAAA+DWEHQAAYGqEHQAAYGqEHQAAYGqEHQeTlJSkRo0aydPTU126dNGuXbtsVvurr75S7969FRISIicnJ33++ec2qy1JM2fOVKdOnVSzZk0FBASoT58+yszMtGkPCxcuVJs2bSwvzIqMjNTatWtt2sPVXn/9dTk5OWns2LE2qzl16lQ5OTlZTS1atLBZ/XKnT5/WH/7wB9WpU0deXl5q3bq19uzZY7P6jRo1qvBzcHJyUnx8vE3ql5aWatKkSQoPD5eXl5eaNGmiGTNm/PZ3AFWxCxcuaOzYsQoLC5OXl5fuvfde7d69u9rq/dbnkGEYmjx5soKDg+Xl5aWoqCgdOXLEZvX/+c9/qmfPnpY37mdkZFRZ7RvpoaSkRBMmTFDr1q3l7e2tkJAQDRkyRGfOnLFZD9LPnxMtWrSQt7e3atWqpaioKO3cubNKe6hKhB0H8vHHHysxMVFTpkzR3r171bZtW0VHRysnJ8cm9S9evKi2bdsqKSnJJvV+aevWrYqPj9eOHTuUkpKikpIS9ezZUxcvXrRZDw0aNNDrr7+u9PR07dmzR927d9fjjz+uAwcO2KyHcrt379bbb7+tNm3a2Lx2q1atdPbsWcv073//26b1z58/r/vuu09ubm5au3atDh48qDfeeEO1atWyWQ+7d++2+hmkpKRIkp566imb1J81a5YWLlyov/3tbzp06JBmzZql2bNna8GCBTapX+6ZZ55RSkqK3n//fe3bt089e/ZUVFSUTp8+XS31futzaPbs2Zo/f74WLVqknTt3ytvbW9HR0bp8+bJN6l+8eFH333+/Zs2aVSX1braHS5cuae/evZo0aZL27t2rf/7zn8rMzNRjjz1msx4k6a677tLf/vY37du3T//+97/VqFEj9ezZUz/88EOV9lFlquLLOFE1OnfubMTHx1vmS0tLjZCQEGPmzJk270WS8dlnn9m87tVycnIMScbWrVvt2ketWrWMv//97zateeHCBaNZs2ZGSkqK8cADDxhjxoyxWe0pU6YYbdu2tVm9a5kwYYJx//3327WHXxozZozRpEkTo6yszCb1YmJijGHDhlkt69u3rxEbG2uT+oZhGJcuXTJcXFyM1atXWy2/5557jJdeeqna6//yc6isrMwICgoy/vKXv1iW5eXlGR4eHsZHH31U7fWvduzYMUOS8Z///KfK695oD+V27dplSDJOnDhhtx7y8/MNScbGjRurpYdbxZkdB1FcXKz09HRFRUVZljk7OysqKkppaWl27Mx+8vPzJUm1a9e2S/3S0lItX75cFy9eVGRkpE1rx8fHKyYmxuq/B1s6cuSIQkJC1LhxY8XGxurkyZM2rb9y5Up17NhRTz31lAICAtS+fXu98847Nu3hasXFxfrggw80bNgwm31Z8L333qvU1FR9++23kqSvv/5a//73v9WrVy+b1JekK1euqLS0VJ6enlbLvby8bH62T5KOHTumrKwsq/9f+Pn5qUuXLnfs56T082elk5OT3b7jsbi4WIsXL5afn5/atm1rlx5+i2neoHy7O3funEpLSyt8xUVgYKAOHz5sp67sp6ysTGPHjtV9992nu+++26a19+3bp8jISF2+fFk+Pj767LPPFBERYbP6y5cv1969e6v1vohf06VLFy1dulTNmzfX2bNnNW3aNP3ud7/T/v37VbNmTZv08N1332nhwoVKTEzUiy++qN27d2v06NFyd3dXXFycTXq42ueff668vDwNHTrUZjVfeOEFFRQUqEWLFnJxcVFpaaleffVVxcbG2qyHmjVrKjIyUjNmzFDLli0VGBiojz76SGlpaWratKnN+iiXlZUlSdf8nCxfd6e5fPmyJkyYoEGDBtn8izlXr16tgQMH6tKlSwoODlZKSorq1q1r0x5uFGEHDik+Pl779++3y2+PzZs3V0ZGhvLz8/Xpp58qLi5OW7dutUngOXXqlMaMGaOUlJQKv03bytVnDtq0aaMuXbooLCxMn3zyiYYPH26THsrKytSxY0e99tprkqT27dtr//79WrRokV3CzpIlS9SrVy+FhITYrOYnn3yiDz/8UMuWLVOrVq2UkZGhsWPHKiQkxKY/g/fff1/Dhg1T/fr15eLionvuuUeDBg1S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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Extract the relevant data\n",
    "data = data_dispatcher[3][0][1]\n",
    "\n",
    "# Count the occurrences of each label\n",
    "label_distribution = Counter(data.numpy())\n",
    "\n",
    "# Plot the data distribution\n",
    "labels, counts = zip(*label_distribution.items())\n",
    "plt.bar(labels, counts, tick_label=labels)\n",
    "plt.xlabel('Labels')\n",
    "plt.ylabel('Count')\n",
    "plt.title('Data Distribution')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "d939520e-80c9-4de8-9b6f-62bad05a5417",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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r+PHj9uNh4WF2a1XBJ8fX18EmIxrdoBbyH//+1/l5+erVatWtrEnnnhC27dv14QJE9S5c2f5+vqqvLxcDz300DXd3v2f//xHvXv3Vtu2bTVv3jw1b95cnp6eWrdunebPn3/dt4j/2P79+yXJrucfCwwMVGZmpjZs2KDPPvtMn332mVJSUjRs2LBKF+5ejSPvTisrK3NYLTc3tyrHLctyWA+AsxF2gFvIO++8I0m2j2vOnz+vtLQ0TZ8+XVOmTLHNO3z4cKXXXu2i2zVr1qi4uFirV6+2O4vw+eef10rPV+Pp6al+/fqpX79+Ki8v1/PPP6833nhDkydPVqtWra7rouFrcfjwYT3wwAO29cLCQp05c0YPP/ywbaxhw4aVHsJYUlKiM2fO2I1dT2/h4eH66quvVF5ebnd259ChQ7btAOzxMRZwi9i8ebNmzpypyMhIDRkyRNL//db/49/yFyxYUOn1Pj4+klTpf95V7SM/P18pKSk33POKFSv0l7/8RdHR0erdu/dV5115m7wkubq62h6cWPEE5qv1X11Lly5VaWmpbX3x4sW6fPmy+vTpYxtr2bKlvvjii0qv+/GZnevp7eGHH1Z2drY++OAD29jly5e1aNEi+fr66r777qvO4QBG48wOYKDPPvtMhw4d0uXLl5WTk6PNmzcrNTVV4eHhWr16tby9vSVJfn5++uUvf6m5c+eqtLRUt912mzZu3KijR49W2meXLl0kSS+//LIGDRokDw8P9evXT7GxsbazKs8995wKCwv15ptvKjAwsNIZjJ/y0UcfydfXVyUlJbYnKG/btk2dOnXSqlWrfvK1Tz/9tM6dO6devXqpWbNmOn78uBYtWqTOnTvbrmXp3Lmz3NzcNGfOHOXn58vLy8v2bKDqKCkpUe/evfXEE08oKytLr7/+uu6991498sgjdn395je/0cCBA/Xggw/qyy+/1IYNG+wenni9vT377LN64403NHz4cGVkZCgiIkIfffSRtm3bpgULFvzktU3ALcvJd4MBqEEVt0VXLJ6enlZwcLD14IMPWq+99prttuQr/fe//7V+9atfWQEBAZa/v7/1+OOPW6dPn67ytumZM2dat912m+Xq6mp3G/rq1autjh07Wt7e3lZERIQ1Z84c66233rrqrepXqrj1vGLx9va2mjVrZvXt29d66623rKKiokqv+fGt5x99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U2xsrGJiYnTq1Klaqfdz70Nz587VwoULtWTJEu3cuVM+Pj6Ki4tTUVGRQ+pfvHhR9957r+bMmVMj9a63h0uXLmnv3r2aPHmy9u7dq7/97W/KysrSI4884rAeJOn222/Xn//8Z+3bt0///Oc/FRERodjYWH377bc12keNqYkv40TN6N69u5WYmGhbLysrs0JDQ63Zs2c7vBdJ1scff+zwulfKzc21JFlbt251ah8NGza0/vKXvzi05oULF6zWrVtbqamp1n333WeNGTPGYbWnTp1qderUyWH1qjJx4kTr3nvvdWoPPzZmzBirZcuWVnl5uUPqxcfHWyNGjLAbGzBggDVkyBCH1Lcsy7p06ZLl5uZmrV271m78rrvusl5++eVar//j96Hy8nIrODjY+sMf/mAby8vLs7y8vKz333+/1utf6ejRo5Yk61//+leN173WHirs2rXLkmQdP37caT3k5+dbkqxNmzbVSg83ijM7dURJSYkyMjIUExNjG3N1dVVMTIzS09Od2Jnz5OfnS5IaNWrklPplZWVauXKlLl68qOjoaIfWTkxMVHx8vN2/D450+PBhhYaGqkWLFhoyZIhOnDjh0PqrV69W165d9fjjjyswMFB33nmn3nzzTYf2cKWSkhK9++67GjFihMO+LPjuu+9WWlqa/v3vf0uSvvzyS/3zn/9Unz59HFJfki5fvqyysjJ5e3vbjderV8/hZ/sk6ejRo8rOzrb778Lf3189evS4Zd8npR/eK11cXJz2HY8lJSVaunSp/P391alTJ6f08HOMeYLyze7s2bMqKyur9BUXQUFBOnTokJO6cp7y8nKNHTtW99xzj+644w6H1t63b5+io6NVVFQkX19fffzxx4qKinJY/ZUrV2rv3r21el3ET+nRo4eWL1+uNm3a6MyZM5o+fbp+8YtfaP/+/WrQoIFDevjmm2+0ePFijR8/Xi+99JJ2796tF154QZ6enkpISHBID1f65JNPlJeXp+HDhzus5osvvqiCggK1bdtWbm5uKisr06xZszRkyBCH9dCgQQNFR0dr5syZateunYKCgvT+++8rPT1drVq1clgfFbKzsyWpyvfJim23mqKiIk2cOFGDBw92+Bdzrl27VoMGDdKlS5cUEhKi1NRUNWnSxKE9XCvCDuqkxMRE7d+/3ym/PbZp00aZmZnKz8/XRx99pISEBG3d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X2rGjBny9fW1Lc8884zOnDljt58Kw4cPV2Zmptq0aaMXXnhBGzdurP0DBVCjCDsAbgrHjh1T37591bFjR/3v//6vMjIylJycLOmH62quVWFhoaZPn67MzEzbsm/fPh0+fFje3t6V5t911106evSoZs6cqe+//15PPPGEHnvssRo7LgC1z93ZDQDAtcjIyFB5ebn+9Kc/ydX1h9/TPvzww0rzLl++rD179qh79+6SpKysLOXl5aldu3aSfggvWVlZatWq1TXX9vPz05NPPqknn3xSjz32mB566CGdO3dOjRo1qoEjA1DbCDsA6pz8/HxlZmbajTVp0kSlpaVatGiR+vXrp23btmnJkiWVXuvh4aHRo0dr4cKFcnd3V1JSknr27GkLP1OmTFHfvn0VFhamxx57TK6urvryyy+1f/9+/e53v6u0v3nz5ikkJER33nmnXF1dtWrVKgUHB9fIwwsBOAYfYwGoc7Zs2aI777zTbnnnnXc0b948zZkzR3fccYfee++9Km8Br1+/viZOnKj/9//+n+655x75+vrqgw8+sG2Pi4vT2rVrtXHjRnXr1k09e/bU/PnzFR4eXmUvDRo00Ny5c9W1a1d169ZNx44d07p162xnlwDUfdyNBQAAjMavJgAAwGiEHQAAYDTCDgAAMBphBwAAGI2wAwAAjEbYAQAARiPsAAAAoxF2AACA0Qg7AADAaIQdAABgNMIOAAAwGmEHAAAY7f8DiFxQJ76EmeAAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Extract the relevant data\n",
    "data = data_dispatcher[6][0][1]\n",
    "\n",
    "# Count the occurrences of each label\n",
    "label_distribution = Counter(data.numpy())\n",
    "\n",
    "# Plot the data distribution\n",
    "labels, counts = zip(*label_distribution.items())\n",
    "plt.bar(labels, counts, tick_label=labels)\n",
    "plt.xlabel('Labels')\n",
    "plt.ylabel('Count')\n",
    "plt.title('Data Distribution')\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9fc5e33d-4e1b-470a-9745-144d6999f1e1",
   "metadata": {},
   "outputs": [],
   "source": [
    "nodes = PENSNode.generate(\n",
    "    data_dispatcher=data_dispatcher,\n",
    "    p2p_net=StaticP2PNetwork(10),\n",
    "    model_proto=TorchModelHandler(\n",
    "        net=model,\n",
    "        optimizer= torch.optim.SGD,\n",
    "        optimizer_params = {\n",
    "            \"lr\": 0.001,\n",
    "            \"momentum\": 0.9,\n",
    "            \"weight_decay\": 5e-4\n",
    "        },\n",
    "        \n",
    "        criterion = F.cross_entropy,\n",
    "        create_model_mode= CreateModelMode.MERGE_UPDATE,\n",
    "        batch_size= 10,\n",
    "        local_epochs= 3),\n",
    "    round_len=100,\n",
    "    sync=False,\n",
    "    n_sampled= 9,\n",
    "    m_top= 9,\n",
    "    step1_rounds= 1)\n",
    "\n",
    "simulator = GossipSimulator(\n",
    "    nodes = nodes,\n",
    "    data_dispatcher=data_dispatcher,\n",
    "    delta=100,\n",
    "    protocol=AntiEntropyProtocol.PUSH,\n",
    "    sampling_eval=1.0\n",
    ")\n",
    "\n",
    "report = SimulationReport()\n",
    "simulator.add_receiver(report)\n",
    "simulator.init_nodes(seed=42)\n",
    "simulator.start(n_rounds=100)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8d902d03-3247-41fc-9db9-3f3d76b223bb",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fe290b96-ed9b-4214-ba0b-41dd3bb9973a",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d04f8524-b454-499b-9ec2-b88e4451e834",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b618a6b5-e2aa-489c-a761-915f75fd2151",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "029c3fc1-9910-4df6-aac9-e55e2301d13a",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dcc0d9c3-a96e-491b-bf4c-39f6ce366c90",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c1fcde59-523f-4216-83bc-e216bc329298",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0005910f-ca5e-4781-bf33-ab9eb9ea0a2c",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "54788e72-35dd-4e9c-b2f7-3b533205dd59",
   "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.10.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
