{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "7ef47ffb-38cd-4095-9561-431be338e1c8",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "import sys\n",
    " \n",
    "# setting path\n",
    "sys.path.append('../')\n",
    "\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_FEMNIST,get_FashionMNIST, get_MNIST\n",
    "from gossipy_original.utils import plot_evaluation\n",
    "\n",
    "\n",
    "    \n",
    "class CNNmnist(TorchModel):\n",
    "    def __init__(self):\n",
    "        super().__init__()\n",
    "        self.conv1 = nn.Conv2d(1, 10, kernel_size=5)\n",
    "        self.act = nn.ReLU()\n",
    "        self.pool = nn.MaxPool2d(2, 2)\n",
    "        self.conv2 = nn.Conv2d(10, 20, kernel_size=5)\n",
    "        self.conv2_drop = nn.Dropout2d()\n",
    "        self.fc1 = nn.Linear(320, 50)\n",
    "        self.fc2 = nn.Linear(50, 10)        \n",
    "        # self.out = nn.Linear(64 * 7 * 7, 10)\n",
    "\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",
    "        pass\n",
    "    \n",
    "    def forward(self, x):\n",
    "        x = x.reshape(-1, 1, 28, 28)\n",
    "        x = F.relu(F.max_pool2d(self.conv1(x), 2))\n",
    "        x = F.relu(F.max_pool2d(self.conv2_drop(self.conv2(x)), 2))\n",
    "        x = x.view(-1, 320)\n",
    "        x = F.relu(self.fc1(x))\n",
    "        x = F.dropout(x, training=self.training)\n",
    "        x = self.fc2(x)\n",
    "        return x\n",
    "    def __repr__(self) -> str:\n",
    "        return \"miniNet(size=%d)\" %self.get_size()\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",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "610d0fa5-0ad8-41e3-8345-59c287c10ef2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([60000])\n",
      "5997\n",
      "[[  0 649]\n",
      " [  1 716]\n",
      " [  2 571]\n",
      " [  3 594]\n",
      " [  4 587]\n",
      " [  5 560]\n",
      " [  6 557]\n",
      " [  7 650]\n",
      " [  8 574]\n",
      " [  9 591]]\n",
      "60000\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "tensor([5, 0, 4,  ..., 5, 6, 8])"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Dataset loading\n",
    "transform = Compose([Normalize([0.5, 0.5, 0.5], [0.5, 0.5, 0.5])])\n",
    "train_set, test_set = get_MNIST()\n",
    "\n",
    "train_set[0].shape\n",
    "\n",
    "tran1=torch.zeros([60000, 28,28], dtype=torch.float32)\n",
    "\n",
    "tran1.shape\n",
    "\n",
    "train_set[1].shape\n",
    "\n",
    "tag1=torch.zeros([60000], dtype=torch.int64)\n",
    "\n",
    "print(tag1.shape)\n",
    "\n",
    "import random\n",
    "list0=[],[]\n",
    "list1=[],[]\n",
    "list2=[],[]\n",
    "list3=[],[]\n",
    "list4=[],[]\n",
    "list5=[],[]\n",
    "list6=[],[]\n",
    "list7=[],[]\n",
    "list8=[],[]\n",
    "list9=[],[]\n",
    "\n",
    "\n",
    "import math\n",
    "import numpy as np\n",
    "for z in range(0,60000,1):\n",
    "    t=math.floor(np.random.uniform(low = 0.0, high = 10.0, size = None))\n",
    "    if t ==0:\n",
    "        list0[0].append(train_set[0][z])\n",
    "        list0[1].append(train_set[1][z])\n",
    "        \n",
    "    elif t ==1:\n",
    "        list1[0].append(train_set[0][z])\n",
    "        list1[1].append(train_set[1][z])\n",
    "\n",
    "    elif t ==2:\n",
    "        list2[0].append(train_set[0][z])\n",
    "        list2[1].append(train_set[1][z])\n",
    "        \n",
    "    elif t ==3:\n",
    "        list3[0].append(train_set[0][z])\n",
    "        list3[1].append(train_set[1][z])        \n",
    "        \n",
    "    elif t ==4:\n",
    "        list4[0].append(train_set[0][z])\n",
    "        list4[1].append(train_set[1][z])        \n",
    "        \n",
    "        \n",
    "    elif t ==5:\n",
    "        list5[0].append(train_set[0][z])\n",
    "        list5[1].append(train_set[1][z])\n",
    "        \n",
    "    elif t ==6:\n",
    "        list6[0].append(train_set[0][z])\n",
    "        list6[1].append(train_set[1][z])        \n",
    "        \n",
    "    elif t ==7:\n",
    "        list7[0].append(train_set[0][z])\n",
    "        list7[1].append(train_set[1][z])   \n",
    "        \n",
    "        \n",
    "    elif t ==8:\n",
    "        list8[0].append(train_set[0][z])\n",
    "        list8[1].append(train_set[1][z])   \n",
    "        \n",
    "    elif t ==9:\n",
    "        list9[0].append(train_set[0][z])\n",
    "        list9[1].append(train_set[1][z])   \n",
    "\n",
    "\n",
    "print(len(list0[0]))\n",
    "\n",
    "unique, counts = np.unique(list9[1], return_counts=True)\n",
    "print(np.asarray((unique, counts)).T)\n",
    "\n",
    "dataset1=[]\n",
    "\n",
    "dataset1.append(list0[0])\n",
    "dataset1.append(list1[0])\n",
    "dataset1.append(list2[0])\n",
    "dataset1.append(list3[0])\n",
    "dataset1.append(list4[0])\n",
    "dataset1.append(list5[0])\n",
    "dataset1.append(list6[0])\n",
    "dataset1.append(list7[0])\n",
    "dataset1.append(list8[0])\n",
    "dataset1.append(list9[0])\n",
    "\n",
    "target1=[]\n",
    "\n",
    "target1.append(list0[1])\n",
    "target1.append(list1[1])\n",
    "target1.append(list2[1])\n",
    "target1.append(list3[1])\n",
    "target1.append(list4[1])\n",
    "target1.append(list5[1])\n",
    "target1.append(list6[1])\n",
    "target1.append(list7[1])\n",
    "target1.append(list8[1])\n",
    "target1.append(list9[1])\n",
    "\n",
    "len(dataset1[1])\n",
    "\n",
    "global o\n",
    "o=0\n",
    "for i in range (len(target1)):\n",
    "    for j in range (len(dataset1[i])):\n",
    "        tran1[o]= dataset1[i][j]\n",
    "        tag1[o]=target1[i][j]\n",
    "        o=o+1\n",
    "print(o)\n",
    "\n",
    "train_set[1]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "0c21f936-247b-47ad-974c-af7e1368a4e1",
   "metadata": {},
   "outputs": [],
   "source": [
    "import random\n",
    "for z in range(0,60000,1):\n",
    "    t=math.floor(np.random.uniform(low = 0.0, high = 10.0, size = None))\n",
    "    if t<5:\n",
    "        pa=random.randint(0,9)\n",
    "        while pa == tag1[z]:\n",
    "            pa=random.randint(0,9)\n",
    "        tag1[z]=pa"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "f87a367e-f092-46c6-aed1-8c340b50791a",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([2, 1, 7,  ..., 8, 7, 8])"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "tag1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "6f48e465-6fd1-42d9-82fe-08c9095db491",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "data_handler = ClassificationDataHandler(tran1, tag1,\n",
    "                                         test_set[0], test_set[1])\n",
    "\n",
    "\n",
    "data_dispatcher = CustomDataDispatcher(data_handler, n=10, eval_on_user=False, auto_assign=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "1e9f9c07-d4f6-4dab-b33e-655da5dbf51a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "this is my idx\n",
      "0\n",
      "this is my n_sampled\n",
      "9\n",
      "this is my m_top\n",
      "9\n",
      "this is my nighbor counter\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my nighbor selected\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my idx\n",
      "1\n",
      "this is my n_sampled\n",
      "9\n",
      "this is my m_top\n",
      "9\n",
      "this is my nighbor counter\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my nighbor selected\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my idx\n",
      "2\n",
      "this is my n_sampled\n",
      "9\n",
      "this is my m_top\n",
      "9\n",
      "this is my nighbor counter\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my nighbor selected\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my idx\n",
      "3\n",
      "this is my n_sampled\n",
      "9\n",
      "this is my m_top\n",
      "9\n",
      "this is my nighbor counter\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my nighbor selected\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my idx\n",
      "4\n",
      "this is my n_sampled\n",
      "9\n",
      "this is my m_top\n",
      "9\n",
      "this is my nighbor counter\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my nighbor selected\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my idx\n",
      "5\n",
      "this is my n_sampled\n",
      "9\n",
      "this is my m_top\n",
      "9\n",
      "this is my nighbor counter\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my nighbor selected\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my idx\n",
      "6\n",
      "this is my n_sampled\n",
      "9\n",
      "this is my m_top\n",
      "9\n",
      "this is my nighbor counter\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my nighbor selected\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my idx\n",
      "7\n",
      "this is my n_sampled\n",
      "9\n",
      "this is my m_top\n",
      "9\n",
      "this is my nighbor counter\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my nighbor selected\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my idx\n",
      "8\n",
      "this is my n_sampled\n",
      "9\n",
      "this is my m_top\n",
      "9\n",
      "this is my nighbor counter\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my nighbor selected\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my idx\n",
      "9\n",
      "this is my n_sampled\n",
      "9\n",
      "this is my m_top\n",
      "9\n",
      "this is my nighbor counter\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n",
      "this is my nighbor selected\n",
      "{0: 0, 1: 0, 2: 0, 3: 0, 4: 0, 5: 0, 6: 0, 7: 0, 8: 0, 9: 0}\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"><span style=\"color: #7fbfbf; text-decoration-color: #7fbfbf\">151023-18:01:58 </span><span style=\"color: #000080; text-decoration-color: #000080\">INFO    </span> Simulation started.                                                           <span style=\"color: #7f7f7f; text-decoration-color: #7f7f7f\">simul.py:372</span>\n",
       "</pre>\n"
      ],
      "text/plain": [
       "\u001b[2;36m151023-18:01:58\u001b[0m\u001b[2;36m \u001b[0m\u001b[34mINFO    \u001b[0m Simulation started.                                                           \u001b[2msimul.py\u001b[0m\u001b[2m:\u001b[0m\u001b[2m372\u001b[0m\n"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"></pre>\n"
      ],
      "text/plain": [
       "\u001b[?25l"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7e462edb9d244d0cbab51141e28a50e2",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "Output()"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\">\n",
       "</pre>\n"
      ],
      "text/plain": [
       "\n",
       "\u001b[?25h"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"><span style=\"color: #7fbfbf; text-decoration-color: #7fbfbf\">151023-20:12:07 </span><span style=\"color: #000080; text-decoration-color: #000080\">INFO    </span> # Sent messages: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">3562</span>                                                         <span style=\"color: #7f7f7f; text-decoration-color: #7f7f7f\">simul.py:239</span>\n",
       "</pre>\n"
      ],
      "text/plain": [
       "\u001b[2;36m151023-20:12:07\u001b[0m\u001b[2;36m \u001b[0m\u001b[34mINFO    \u001b[0m # Sent messages: \u001b[1;36m3562\u001b[0m                                                         \u001b[2msimul.py\u001b[0m\u001b[2m:\u001b[0m\u001b[2m239\u001b[0m\n"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"><span style=\"color: #7fbfbf; text-decoration-color: #7fbfbf\">                </span><span style=\"color: #000080; text-decoration-color: #000080\">INFO    </span> # Failed messages: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">0</span>                                                          <span style=\"color: #7f7f7f; text-decoration-color: #7f7f7f\">simul.py:240</span>\n",
       "</pre>\n"
      ],
      "text/plain": [
       "\u001b[2;36m               \u001b[0m\u001b[2;36m \u001b[0m\u001b[34mINFO    \u001b[0m # Failed messages: \u001b[1;36m0\u001b[0m                                                          \u001b[2msimul.py\u001b[0m\u001b[2m:\u001b[0m\u001b[2m240\u001b[0m\n"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "text/html": [
       "<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"><span style=\"color: #7fbfbf; text-decoration-color: #7fbfbf\">                </span><span style=\"color: #000080; text-decoration-color: #000080\">INFO    </span> Total size: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">77794080</span>                                                          <span style=\"color: #7f7f7f; text-decoration-color: #7f7f7f\">simul.py:241</span>\n",
       "</pre>\n"
      ],
      "text/plain": [
       "\u001b[2;36m               \u001b[0m\u001b[2;36m \u001b[0m\u001b[34mINFO    \u001b[0m Total size: \u001b[1;36m77794080\u001b[0m                                                          \u001b[2msimul.py\u001b[0m\u001b[2m:\u001b[0m\u001b[2m241\u001b[0m\n"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[{'accuracy': 0.19116}, {'accuracy': 0.40121}, {'accuracy': 0.5760399999999999}, {'accuracy': 0.7459100000000001}, {'accuracy': 0.68541}, {'accuracy': 0.8411500000000001}, {'accuracy': 0.8499400000000001}, {'accuracy': 0.8721}, {'accuracy': 0.8838900000000001}, {'accuracy': 0.8861600000000001}, {'accuracy': 0.90601}, {'accuracy': 0.90628}, {'accuracy': 0.91272}, {'accuracy': 0.91676}, {'accuracy': 0.9206499999999999}, {'accuracy': 0.9247299999999999}, {'accuracy': 0.9281199999999998}, {'accuracy': 0.92867}, {'accuracy': 0.9295599999999998}, {'accuracy': 0.93435}, {'accuracy': 0.9324200000000001}, {'accuracy': 0.9337099999999999}, {'accuracy': 0.9384}, {'accuracy': 0.9398399999999999}, {'accuracy': 0.94086}, {'accuracy': 0.9420200000000001}, {'accuracy': 0.94512}, {'accuracy': 0.9447700000000001}, {'accuracy': 0.9447800000000001}, {'accuracy': 0.94716}, {'accuracy': 0.9500500000000001}, {'accuracy': 0.9495799999999999}, {'accuracy': 0.9499000000000001}, {'accuracy': 0.95175}, {'accuracy': 0.9517}, {'accuracy': 0.9533799999999999}, {'accuracy': 0.9522499999999999}, {'accuracy': 0.95543}, {'accuracy': 0.9548}, {'accuracy': 0.95604}, {'accuracy': 0.9559799999999999}, {'accuracy': 0.95687}, {'accuracy': 0.9577199999999999}, {'accuracy': 0.95731}, {'accuracy': 0.9575800000000001}, {'accuracy': 0.9590099999999999}, {'accuracy': 0.9578999999999999}, {'accuracy': 0.96013}, {'accuracy': 0.9606}, {'accuracy': 0.9610999999999998}, {'accuracy': 0.9608099999999998}, {'accuracy': 0.9609}, {'accuracy': 0.9612999999999999}, {'accuracy': 0.9627899999999998}, {'accuracy': 0.96264}, {'accuracy': 0.96295}, {'accuracy': 0.96259}, {'accuracy': 0.9639599999999999}, {'accuracy': 0.96372}, {'accuracy': 0.9640600000000001}, {'accuracy': 0.96379}, {'accuracy': 0.9647400000000002}, {'accuracy': 0.9646000000000001}, {'accuracy': 0.9641}, {'accuracy': 0.96494}, {'accuracy': 0.96477}, {'accuracy': 0.96573}, {'accuracy': 0.9655999999999999}, {'accuracy': 0.9659599999999999}, {'accuracy': 0.9660400000000001}, {'accuracy': 0.9664699999999999}, {'accuracy': 0.966}, {'accuracy': 0.96587}, {'accuracy': 0.96639}, {'accuracy': 0.9665700000000002}, {'accuracy': 0.9666499999999999}, {'accuracy': 0.96624}, {'accuracy': 0.9663700000000001}, {'accuracy': 0.9676899999999999}, {'accuracy': 0.9677199999999999}, {'accuracy': 0.9677000000000001}, {'accuracy': 0.96771}, {'accuracy': 0.9686299999999999}, {'accuracy': 0.9688399999999999}, {'accuracy': 0.96782}, {'accuracy': 0.96832}, {'accuracy': 0.9687999999999999}, {'accuracy': 0.96893}, {'accuracy': 0.96893}, {'accuracy': 0.9689300000000001}, {'accuracy': 0.9695199999999999}, {'accuracy': 0.9696800000000001}, {'accuracy': 0.9689500000000001}, {'accuracy': 0.9690299999999998}, {'accuracy': 0.96975}, {'accuracy': 0.96921}, {'accuracy': 0.9696100000000001}, {'accuracy': 0.9700799999999999}, {'accuracy': 0.9694499999999999}, {'accuracy': 0.9698500000000001}, {'accuracy': 0.9702500000000001}, {'accuracy': 0.97}, {'accuracy': 0.97041}, {'accuracy': 0.97048}, {'accuracy': 0.97065}, {'accuracy': 0.9718399999999999}, {'accuracy': 0.9711299999999999}, {'accuracy': 0.97097}, {'accuracy': 0.9709}, {'accuracy': 0.9711699999999999}, {'accuracy': 0.9713799999999999}, {'accuracy': 0.97149}, {'accuracy': 0.9720500000000001}, {'accuracy': 0.97172}, {'accuracy': 0.9714199999999998}, {'accuracy': 0.97173}, {'accuracy': 0.97159}, {'accuracy': 0.9716199999999999}, {'accuracy': 0.9719300000000001}, {'accuracy': 0.97136}, {'accuracy': 0.97255}, {'accuracy': 0.97269}, {'accuracy': 0.9724999999999999}, {'accuracy': 0.9722}, {'accuracy': 0.97284}, {'accuracy': 0.9725100000000001}, {'accuracy': 0.97262}, {'accuracy': 0.97255}, {'accuracy': 0.9730400000000001}, {'accuracy': 0.97326}, {'accuracy': 0.97352}, {'accuracy': 0.9728199999999998}, {'accuracy': 0.9732800000000001}, {'accuracy': 0.97279}, {'accuracy': 0.97369}, {'accuracy': 0.97306}, {'accuracy': 0.9739000000000001}, {'accuracy': 0.9737}, {'accuracy': 0.9737800000000002}, {'accuracy': 0.9745699999999999}, {'accuracy': 0.9739699999999999}, {'accuracy': 0.97413}, {'accuracy': 0.9745499999999998}, {'accuracy': 0.9749099999999998}, {'accuracy': 0.9747999999999999}, {'accuracy': 0.9745700000000002}, {'accuracy': 0.9748299999999999}, {'accuracy': 0.9742}, {'accuracy': 0.9738199999999999}, {'accuracy': 0.97409}, {'accuracy': 0.8949999999999999}, {'accuracy': 0.96595}, {'accuracy': 0.96936}, {'accuracy': 0.96722}, {'accuracy': 0.9692000000000001}, {'accuracy': 0.96926}, {'accuracy': 0.97033}, {'accuracy': 0.97021}, {'accuracy': 0.9697000000000001}, {'accuracy': 0.97058}, {'accuracy': 0.9712999999999999}, {'accuracy': 0.97129}, {'accuracy': 0.97179}, {'accuracy': 0.9723799999999999}, {'accuracy': 0.9714799999999999}, {'accuracy': 0.9715299999999999}, {'accuracy': 0.9714699999999998}, {'accuracy': 0.9728599999999998}, {'accuracy': 0.97223}, {'accuracy': 0.9724799999999998}, {'accuracy': 0.97197}, {'accuracy': 0.9729599999999999}, {'accuracy': 0.97387}, {'accuracy': 0.9732700000000001}, {'accuracy': 0.9739799999999998}, {'accuracy': 0.9728900000000001}, {'accuracy': 0.9745799999999999}, {'accuracy': 0.9734}, {'accuracy': 0.9730000000000001}, {'accuracy': 0.97423}, {'accuracy': 0.9743399999999998}, {'accuracy': 0.9726000000000001}, {'accuracy': 0.9744400000000001}, {'accuracy': 0.9732899999999999}, {'accuracy': 0.9753000000000001}, {'accuracy': 0.97388}, {'accuracy': 0.9723499999999999}, {'accuracy': 0.97478}, {'accuracy': 0.97547}, {'accuracy': 0.9745900000000001}, {'accuracy': 0.9753200000000002}, {'accuracy': 0.97428}, {'accuracy': 0.9752599999999999}, {'accuracy': 0.97438}, {'accuracy': 0.9746600000000001}, {'accuracy': 0.97495}, {'accuracy': 0.97625}, {'accuracy': 0.9747399999999999}, {'accuracy': 0.9761200000000001}, {'accuracy': 0.9740399999999999}, {'accuracy': 0.97187}, {'accuracy': 0.97513}, {'accuracy': 0.9763199999999997}, {'accuracy': 0.97417}, {'accuracy': 0.9752899999999999}, {'accuracy': 0.9748000000000001}, {'accuracy': 0.97568}, {'accuracy': 0.97631}, {'accuracy': 0.97424}, {'accuracy': 0.9754200000000001}, {'accuracy': 0.97654}, {'accuracy': 0.97691}, {'accuracy': 0.97427}, {'accuracy': 0.9766799999999998}, {'accuracy': 0.97711}, {'accuracy': 0.9767899999999999}, {'accuracy': 0.9760900000000001}, {'accuracy': 0.9721799999999998}, {'accuracy': 0.97606}, {'accuracy': 0.9747399999999999}, {'accuracy': 0.97488}, {'accuracy': 0.97528}, {'accuracy': 0.9743999999999998}, {'accuracy': 0.9748299999999999}, {'accuracy': 0.97503}, {'accuracy': 0.9758899999999999}, {'accuracy': 0.97449}, {'accuracy': 0.9764100000000001}, {'accuracy': 0.97704}, {'accuracy': 0.97675}, {'accuracy': 0.9763399999999999}, {'accuracy': 0.97607}, {'accuracy': 0.9772000000000001}, {'accuracy': 0.9747199999999999}, {'accuracy': 0.976}, {'accuracy': 0.9771000000000001}, {'accuracy': 0.97585}, {'accuracy': 0.97722}, {'accuracy': 0.97552}, {'accuracy': 0.97736}, {'accuracy': 0.9763399999999999}, {'accuracy': 0.9758699999999999}, {'accuracy': 0.97624}, {'accuracy': 0.9759300000000002}, {'accuracy': 0.9759800000000001}, {'accuracy': 0.9762700000000001}, {'accuracy': 0.9774800000000001}, {'accuracy': 0.97507}, {'accuracy': 0.97644}, {'accuracy': 0.9760099999999999}, {'accuracy': 0.97521}, {'accuracy': 0.9765}, {'accuracy': 0.9749599999999999}, {'accuracy': 0.97521}, {'accuracy': 0.97711}, {'accuracy': 0.97683}, {'accuracy': 0.9770300000000001}, {'accuracy': 0.9776300000000001}, {'accuracy': 0.9773700000000002}, {'accuracy': 0.97674}, {'accuracy': 0.97669}, {'accuracy': 0.9768399999999999}, {'accuracy': 0.9775499999999999}, {'accuracy': 0.97539}, {'accuracy': 0.9755400000000002}, {'accuracy': 0.97669}, {'accuracy': 0.9776}, {'accuracy': 0.9778500000000001}, {'accuracy': 0.9780600000000002}, {'accuracy': 0.97507}, {'accuracy': 0.97783}, {'accuracy': 0.97743}, {'accuracy': 0.97676}, {'accuracy': 0.97773}, {'accuracy': 0.9780099999999999}, {'accuracy': 0.9749199999999998}, {'accuracy': 0.9767700000000001}, {'accuracy': 0.9768100000000001}, {'accuracy': 0.9767999999999999}, {'accuracy': 0.97693}, {'accuracy': 0.97682}, {'accuracy': 0.9765099999999999}, {'accuracy': 0.9756699999999998}, {'accuracy': 0.9771500000000002}, {'accuracy': 0.9764800000000001}, {'accuracy': 0.97679}, {'accuracy': 0.97651}, {'accuracy': 0.97635}, {'accuracy': 0.97524}, {'accuracy': 0.97821}, {'accuracy': 0.9761999999999998}, {'accuracy': 0.97638}, {'accuracy': 0.97683}, {'accuracy': 0.9760099999999999}, {'accuracy': 0.9780099999999999}, {'accuracy': 0.9765600000000001}, {'accuracy': 0.9773}, {'accuracy': 0.9772400000000001}, {'accuracy': 0.9781299999999999}, {'accuracy': 0.97707}]\n"
     ]
    }
   ],
   "source": [
    "\n",
    "nodes = PENSNode.generate(\n",
    "    data_dispatcher=data_dispatcher,\n",
    "    p2p_net=StaticP2PNetwork(10),\n",
    "    model_proto=TorchModelHandler(\n",
    "        net=CNNmnist(),\n",
    "        optimizer= torch.optim.SGD,\n",
    "        optimizer_params = {\n",
    "            \"lr\": 0.01,\n",
    "            \"weight_decay\": 0.001\n",
    "        },\n",
    "        criterion = F.cross_entropy, \n",
    "        #loss function\n",
    "        create_model_mode= CreateModelMode.MERGE_UPDATE,\n",
    "        batch_size= 50,\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\n",
    ")\n",
    "\n",
    "report = SimulationReport()\n",
    "simulator.add_receiver(report)\n",
    "simulator.init_nodes(seed=50)\n",
    "simulator.start(n_rounds=300)\n",
    "\n",
    "print([ev for _, ev in report.get_evaluation(False)])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "054e7a28-741d-4173-92dc-00b98386b953",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"><span style=\"color: #7fbfbf; text-decoration-color: #7fbfbf\">                </span><span style=\"color: #000080; text-decoration-color: #000080\">INFO    </span> accuracy: <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">0.98</span>                                                                <span style=\"color: #7f7f7f; text-decoration-color: #7f7f7f\">utils.py:172</span>\n",
       "</pre>\n"
      ],
      "text/plain": [
       "\u001b[2;36m               \u001b[0m\u001b[2;36m \u001b[0m\u001b[34mINFO    \u001b[0m accuracy: \u001b[1;36m0.98\u001b[0m                                                                \u001b[2mutils.py\u001b[0m\u001b[2m:\u001b[0m\u001b[2m172\u001b[0m\n"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjcAAAHHCAYAAABDUnkqAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjYuMywgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/P9b71AAAACXBIWXMAAA9hAAAPYQGoP6dpAABc40lEQVR4nO3deVyU1f4H8M/MMAvDMuyrKArmkgqK6aU0TVFK85ptZt00SrPMsmi1UlpuYXX1p6VleVPbtby2l2Yklkq5524uIC6sIgwMMOv5/YGMTqAyOjMPy+f9es1LeeZZvvMIni/n+Z5zZEIIASIiIqJWQi51AERERESuxOSGiIiIWhUmN0RERNSqMLkhIiKiVoXJDREREbUqTG6IiIioVWFyQ0RERK0KkxsiIiJqVZjcEBERUavC5IaI3Co7OxsymQzZ2dn2bffccw9iY2Mli6k1eeGFFyCTyaQOg6hZYXJD1Mzt2bMH//rXvxAdHQ21Wo2oqCjcdddd2LNnj9Shud3bb7+NpUuXuvUae/fuxQsvvIC8vDy3XseTXn31VXz11VdSh0EkGSY3RM3YypUr0adPH2RlZSEtLQ1vv/027rvvPqxduxZ9+vTBl19+KXWIbuWp5ObFF19kckPUinhJHQARNe7w4cO4++670alTJ/z6668IDQ21vzdt2jQMHDgQd999N3bu3IlOnTp5LC6DwQAfHx+PXa8l4D0hal7Yc0PUTL3xxhuorq7Ge++955DYAEBISAjeffddGAwGvP766wCAFStWQCaTYd26dQ3O9e6770Imk2H37t32bfv378ett96KoKAgaDQa9O3bF998843DcUuXLrWfc8qUKQgLC0O7du0AAEePHsWUKVPQpUsXeHt7Izg4GLfddpvLekBiY2OxZ88erFu3DjKZDDKZDIMHD7a/X15ejkcffRQxMTFQq9WIj4/Ha6+9BpvN5nCeZcuWISkpCX5+fvD390fPnj0xb948++e77bbbAADXXXed/Trn1gf93T333ANfX18cPnwYI0aMgJ+fH+666y4AgM1mw9y5c3HllVdCo9EgPDwckydPxunTpx3OsWXLFqSmpiIkJATe3t7o2LEj7r33Xvv7jdUpAUBeXh5kMtkFe7NkMhkMBgM++OAD++e55557AACVlZV49NFHERsbC7VajbCwMAwbNgzbtm077/mIWiL23BA1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fPDBB7jrrrvQs2dP3HfffejYsSPy8vLw/vvvo7S0FJ999pl9mG89pVKJm2++GcuWLYPBYMB//vOfBudesGABBgwYgJ49e2LSpEno1KkTioqKkJOTg+PHj+PPP/+8aHw33ngjPvroI+h0OnTv3h05OTn4+eefERwc7LJ7kJSUhHfeeQf//ve/ER8fj7CwMAwZMgRPPvkkvvnmG9x444245557kJSUBIPBgF27dmHFihXIy8tDSEgIJk6ciLKyMgwZMgTt2rXD0aNH8dZbbyExMdE+3DsxMREKhQKvvfYaKioqoFarMWTIEISFhTkV66BBgzB58mRkZmZix44dGD58OJRKJQ4ePIgvvvgC8+bNw6233ooPPvgAb7/9NsaMGYO4uDhUVlZi0aJF8Pf3x4gRIwDUPZa87bbb8NZbb0EmkyEuLg7fffddg1qoC923n3/+GXPmzEFUVBQ6duyILl26oF27drj11luRkJAAX19f/Pzzz9i8ebND7xtRqyDtYC0iupidO3eKcePGicjISKFUKkVERIQYN26c2LVr13mPWbNmjQAgZDKZOHbsWKP7HD58WIwfP15EREQIpVIpoqOjxY033ihWrFhh36d+KHhjw5ZPnz4t0tLSREhIiPD19RWpqali//79okOHDg7DkC9nKHhhYaEYOXKk8PPzEwAchoVXVlaK6dOni/j4eKFSqURISIi4+uqrxX/+8x9hMpmEEEKsWLFCDB8+XISFhQmVSiXat28vJk+eLAoKChyus2jRItGpUyehUCguOix8woQJwsfH57zvv/feeyIpKUl4e3sLPz8/0bNnT/HUU0+JkydPCiGE2LZtmxg3bpxo3769UKvVIiwsTNx4441iy5YtDucpKSkRt9xyi9BqtSIwMFBMnjxZ7N69u0lDwffv3y+uvfZa4e3tLQCICRMmCKPRKJ588kmRkJAg/Pz8hI+Pj0hISBBvv/32hf4JiFokmRB/6x8lIiIiasFYc0NEREStCpMbIiIialWY3BAREVGrwuSGiIiIWhUmN0RERNSqMLkhIiKiVqXNTeJns9lw8uRJ+Pn5uWzadSIiInIvIQQqKysRFRXVYG23v2tzyc3JkycRExMjdRhERER0CY4dO4Z27dpdcJ82l9z4+fkBqLs59SspExERUfOm1+sRExNjb8cvpM0lN/WPovz9/ZncEBERtTBNKSmR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eUGfR/2imRwGTkTUfNSYreg+c7Uk1977Uiq0qqY3s7fccovD14sXL0ZoaCj27t2LjRs3oqSkBJs3b7a3d/Hx8fZ9X3nlFdxxxx148cUX7dsSEhKcjvnRRx/FzTff7LDtiSeesP/94YcfxurVq/H555+jX79+qKysxLx58zB//nxMmDABABAXF4cBAwYAAG6++WZMnToVX3/9NW6//XYAwNKlS3HPPfe4dYJFSXtuSktLYbVa7RlevfDwcBQWFjbpHE8//TSioqKQkpLS6PuZmZnQ6XT2V0xMzGXH3RiLtS57ZjExERFdioMHD2LcuHHo1KkT/P39ERsbCwDIz8/Hjh070Lt37/P+Ir9jxw4MHTr0smPo27evw9dWqxUvv/wyevbsiaCgIPj6+mL16tXIz88HAOzbtw9Go/G819ZoNLj77rvtT2O2bduG3bt345577rnsWC+kRc9QPGvWLCxbtgzZ2dnQaDSN7jN9+nSkp6fbv9br9W5JcCz2x1KSP+kjIqIzvJUK7H0pVbJrO2PUqFHo0KEDFi1ahKioKNhsNvTo0QMmkwne3t4XvtZF3pfJZA0ekzVWMOzj4+Pw9RtvvIF58+Zh7ty56NmzJ3x8fPDoo4/CZDI16bpA3aOpxMREHD9+HEuWLMGQIUPQoUOHix53OSRNbkJCQqBQKFBUVOSwvaioCBERERc89j//+Q9mzZqFn3/+Gb169Trvfmq1Gmq12iXxXsjZRTPZc0NE1FzIZDKnHg1J5dSpUzhw4IB9RDAArF+/3v5+r1698N///hdlZWWN9t706tULWVlZSEtLa/T8oaGhKCgosH998ODBJi0uumHDBowePRr/+te/ANQVD//111/o3r07AKBz587w9vZGVlYWJk6c2Og5evbsib59+2LRokX49NNPMX/+/Ite93JJ2s2gUqmQlJTkUAxcXxycnJx83uNef/11vPzyy1i1alWDLjSpWFhQTERElygwMBDBwcF47733cOjQIfzyyy8OTx3GjRuHiIgI3HTTTdiwYQOOHDmC//3vf/b61IyMDHz22WfIyMjAvn37sGvXLrz22mv244cMGYL58+dj+/bt2LJlCx544AEolcqLxtW5c2esWbMGGzduxL59+zB58mSHDgmNRoOnn34aTz31FD788EMcPnwYv//+O95//32H80ycOBGzZs2CEMJhFJe7SP4M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wYDk5ORm+X3V5hbOdKXS0lJYrVaHfywACA8Px/79+yWKqnno378/li5dii5duqCgoAAvvvgiBg4ciN27d6OwsBAqlQoBAQEOx4SHh6OwsFCagJuJ+s/f2PdU/XuFhYUICwtzeN/LywtBQUFt8v5df/31uPnmm9GxY0ccPnwYzz77LG644Qbk5ORAoVC02ftls9nw6KOP4pprrkGPHj0AoEk/e4WFhY1+/9W/1xo1dq8A4M4770SHDh0QFRWFnTt34umnn8aBAwewcuVKAG3vXu3atQvJycmora2Fr68vvvzyS3Tv3h07duxodt9XTG7ILW644Qb733v16oX+/fujQ4cO+Pzzz+Ht7S1hZNTa3HHHHfa/9+zZE7169UJcXByys7MxdOhQCSOT1kMPPYTdu3c71LpR4853r86ty+rZsyciIyMxdOhQHD58GHFxcZ4OU3JdunTBjh07UFFRgRUrVmDChAlYt26d1GE1io+lLkNISAgUCkWDivCioiJERERIFFXzFBAQgCuuuAKHDh1CREQETCYTysvLHfbhfYP981/oeyoiIqJBwbrFYkFZWVmbv38A0KlTJ4SEhODQoUMA2ub9mjp1Kr777jusXbsW7dq1s29vys9eREREo99/9e+1Nue7V43p378/ADh8b7Wle6VSqRAfH4+kpCRkZmYiISEB8+bNa5bfV0xuLoNKpUJSUhKysrLs22w2G7KyspCcnCxhZM1PVVUVDh8+jMjISCQlJUGpVDrctwMHDiA/P7/N37eOHTsiIiLC4d7o9Xr88ccf9nuTnJyM8vJybN261b7PL7/8ApvNZv/Pty07fvw4Tp06hcjISABt634JITB16lR8+eWX+OWXX9CxY0eH95vys5ecnIxdu3Y5JIRr1qyBv78/unfv7pkP4gEXu1eN2bFjBwA4fG+1hXt1PjabDUajsXl+X7m8RLmNWbZsmVCr1WLp0qVi79694v777xcBAQEOFeFt0eOPPy6ys7NFbm6u2LBhg0hJSREhISGiuLhYCCHEAw88INq3by9++eUXsWXLFpGcnCySk5MljtozKisrxfbt28X27dsFADFnzhyxfft2cfToUSGEELNmzRIBAQHi66+/Fjt37hSjR48WHTt2FDU1NfZzXH/99aJ3797ijz/+EOvXrxedO3cW48aNk+ojudWF7ldlZaV44oknRE5OjsjNzRU///yz6NOnj+jcubOora21n6Ot3K8HH3xQ6HQ6kZ2dLQoKCuyv6upq+z4X+9mzWCyiR48eYvjw4WLHjh1i1apVIjQ0VEyfPl2Kj+Q2F7tXhw4dEi+99JLYsmWLyM3NFV9//bXo1KmTuPbaa+3naCv3SgghnnnmGbFu3TqRm5srdu7cKZ555hkhk8nETz/9JIRoft9XTG5c4K233hLt27cXKpVK9OvXT/z+++9ShyS5sWPHisjISKFSqUR0dLQYO3asOHTokP39mpoaMWXKFBEYGCi0Wq0YM2aMKCgokDBiz1m7dq0A0OA1YcIEIUTdcPAZM2aI8PBwoVarxdChQ8WBAwccznHq1Ckxbtw44evrK/z9/UVaWpqorKyU4NO434XuV3V1tRg+fLgIDQ0VSqVSdOjQQUyaNKnBLxdt5X41dp8AiCVLltj3acrPXl5enrjhhhuEt7e3CAkJEY8//rgwm80e/jTudbF7lZ+fL6699loRFBQk1Gq1iI+PF08++aSoqKhwOE9buFdCCHHvvfeKDh06CJVKJUJDQ8XQoUPtiY0Qze/7SiaEEK7vDyIiIiKSBmtuiIiIqFVhckNEREStCpMbIiIialWY3BAREVGrwuSGiIiIWhUmN0RERNSqMLkhIiKiVoXJDRG1CdnZ2ZDJZA3WvyGi1ofJDREREbUqTG6IiIioVWFyQ0TNis1mw+uvv474+Hio1Wq0b98er7zyCoYMGYKpU6c67FtSUgKVSmVfjdhoNOLpp59GTEwM1Go14uPj8f7775/3WuvXr8fAgQPh7e2NmJgYPPLIIzAYDG79fETkfkxuiKhZmT59OmbNmoUZM2Zg7969+PTTTxEeHo6JEyfi008/hdFotO/78ccfIzo6GkOGDAEAjB8/Hp999hnefPNN7Nu3D++++y58fX0bvc7hw4dx/fXX45ZbbsHOnTuxfPlyrF+/vkECRUQtDxfOJKJmo7KyEqGhoZg/fz4mTpzo8F5tbS2ioqKwcOFC3H777QCAhIQE3HzzzcjIyMBff/2FLl26YM2aNUhJSWlw7uzsbFx33XU4ffo0AgICMHHiRCgUCrz77rv2fdavX49BgwbBYDBAo9G498MSkduw54aImo19+/bBaDRi6NChDd7TaDS4++67sXjxYgDAtm3bsHv3btxzzz0AgB07dkChUGDQoEFNutaff/6JpUuXwtfX1/5KTU2FzWZDbm6uyz4TEXmel9QBEBHV8/b2vuD7EydORGJiIo4fP44lS5ZgyJAh6NChQ5OO/buqqipMnjwZjzzySIP32rdv79S5iKh5Yc8NETUbnTt3hre3t71A+O969uyJvn37YtGiRfj0009x7733Orxns9mwbt26Jl2rT58+2Lt3L+Lj4xu8VCqVSz4PEUmDyQ0RNRsajQZPP/00nnrqKXz44Yc4fPgwfv/9d4cRTxMnTsSsWbMghMCYMWPs22NjYzFhwgTce++9+Oqrr5Cbm4vs7Gx8/vnnjV7r6aefxsaNGzF16lTs2LEDBw8exNdff82CYqJWgMkNETUrM2bMwOOPP46ZM2eiW7duGDt2LIqLi+3vjxs3Dl5eXhg3blyDot933nkHt956K6ZMmYKuXbti0qRJ5x3a3atXL6xbtw5//fUXBg4ciN69e2PmzJmIiopy6+cjIvfjaCkialHy8vIQFxeHzZs3o0+fPlKHQ0TNEJMbImoRzGYzTp06hSeeeAK5ubnYsGGD1CERUTPFx1JE1CJs2LABkZGR2Lx5MxYuXCh1OETUjLHnhoiIiFoV9twQERFRq8LkhoiIiFoVJjdERETUqjC5ISIiolaFyQ0RERG1KkxuiIiIqFVhckNEREStCpMbIiIialWY3BAREVGr8v/J/orPEkXqMwAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plot_evaluation([[ev for _, ev in report.get_evaluation(False)]], \"Overall test results\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "c137896b-470c-4faf-aebe-496ca123f991",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[{'accuracy': 0.19116}, {'accuracy': 0.40121}, {'accuracy': 0.5760399999999999}, {'accuracy': 0.7459100000000001}, {'accuracy': 0.68541}, {'accuracy': 0.8411500000000001}, {'accuracy': 0.8499400000000001}, {'accuracy': 0.8721}, {'accuracy': 0.8838900000000001}, {'accuracy': 0.8861600000000001}, {'accuracy': 0.90601}, {'accuracy': 0.90628}, {'accuracy': 0.91272}, {'accuracy': 0.91676}, {'accuracy': 0.9206499999999999}, {'accuracy': 0.9247299999999999}, {'accuracy': 0.9281199999999998}, {'accuracy': 0.92867}, {'accuracy': 0.9295599999999998}, {'accuracy': 0.93435}, {'accuracy': 0.9324200000000001}, {'accuracy': 0.9337099999999999}, {'accuracy': 0.9384}, {'accuracy': 0.9398399999999999}, {'accuracy': 0.94086}, {'accuracy': 0.9420200000000001}, {'accuracy': 0.94512}, {'accuracy': 0.9447700000000001}, {'accuracy': 0.9447800000000001}, {'accuracy': 0.94716}, {'accuracy': 0.9500500000000001}, {'accuracy': 0.9495799999999999}, {'accuracy': 0.9499000000000001}, {'accuracy': 0.95175}, {'accuracy': 0.9517}, {'accuracy': 0.9533799999999999}, {'accuracy': 0.9522499999999999}, {'accuracy': 0.95543}, {'accuracy': 0.9548}, {'accuracy': 0.95604}, {'accuracy': 0.9559799999999999}, {'accuracy': 0.95687}, {'accuracy': 0.9577199999999999}, {'accuracy': 0.95731}, {'accuracy': 0.9575800000000001}, {'accuracy': 0.9590099999999999}, {'accuracy': 0.9578999999999999}, {'accuracy': 0.96013}, {'accuracy': 0.9606}, {'accuracy': 0.9610999999999998}, {'accuracy': 0.9608099999999998}, {'accuracy': 0.9609}, {'accuracy': 0.9612999999999999}, {'accuracy': 0.9627899999999998}, {'accuracy': 0.96264}, {'accuracy': 0.96295}, {'accuracy': 0.96259}, {'accuracy': 0.9639599999999999}, {'accuracy': 0.96372}, {'accuracy': 0.9640600000000001}, {'accuracy': 0.96379}, {'accuracy': 0.9647400000000002}, {'accuracy': 0.9646000000000001}, {'accuracy': 0.9641}, {'accuracy': 0.96494}, {'accuracy': 0.96477}, {'accuracy': 0.96573}, {'accuracy': 0.9655999999999999}, {'accuracy': 0.9659599999999999}, {'accuracy': 0.9660400000000001}, {'accuracy': 0.9664699999999999}, {'accuracy': 0.966}, {'accuracy': 0.96587}, {'accuracy': 0.96639}, {'accuracy': 0.9665700000000002}, {'accuracy': 0.9666499999999999}, {'accuracy': 0.96624}, {'accuracy': 0.9663700000000001}, {'accuracy': 0.9676899999999999}, {'accuracy': 0.9677199999999999}, {'accuracy': 0.9677000000000001}, {'accuracy': 0.96771}, {'accuracy': 0.9686299999999999}, {'accuracy': 0.9688399999999999}, {'accuracy': 0.96782}, {'accuracy': 0.96832}, {'accuracy': 0.9687999999999999}, {'accuracy': 0.96893}, {'accuracy': 0.96893}, {'accuracy': 0.9689300000000001}, {'accuracy': 0.9695199999999999}, {'accuracy': 0.9696800000000001}, {'accuracy': 0.9689500000000001}, {'accuracy': 0.9690299999999998}, {'accuracy': 0.96975}, {'accuracy': 0.96921}, {'accuracy': 0.9696100000000001}, {'accuracy': 0.9700799999999999}, {'accuracy': 0.9694499999999999}, {'accuracy': 0.9698500000000001}, {'accuracy': 0.9702500000000001}, {'accuracy': 0.97}, {'accuracy': 0.97041}, {'accuracy': 0.97048}, {'accuracy': 0.97065}, {'accuracy': 0.9718399999999999}, {'accuracy': 0.9711299999999999}, {'accuracy': 0.97097}, {'accuracy': 0.9709}, {'accuracy': 0.9711699999999999}, {'accuracy': 0.9713799999999999}, {'accuracy': 0.97149}, {'accuracy': 0.9720500000000001}, {'accuracy': 0.97172}, {'accuracy': 0.9714199999999998}, {'accuracy': 0.97173}, {'accuracy': 0.97159}, {'accuracy': 0.9716199999999999}, {'accuracy': 0.9719300000000001}, {'accuracy': 0.97136}, {'accuracy': 0.97255}, {'accuracy': 0.97269}, {'accuracy': 0.9724999999999999}, {'accuracy': 0.9722}, {'accuracy': 0.97284}, {'accuracy': 0.9725100000000001}, {'accuracy': 0.97262}, {'accuracy': 0.97255}, {'accuracy': 0.9730400000000001}, {'accuracy': 0.97326}, {'accuracy': 0.97352}, {'accuracy': 0.9728199999999998}, {'accuracy': 0.9732800000000001}, {'accuracy': 0.97279}, {'accuracy': 0.97369}, {'accuracy': 0.97306}, {'accuracy': 0.9739000000000001}, {'accuracy': 0.9737}, {'accuracy': 0.9737800000000002}, {'accuracy': 0.9745699999999999}, {'accuracy': 0.9739699999999999}, {'accuracy': 0.97413}, {'accuracy': 0.9745499999999998}, {'accuracy': 0.9749099999999998}, {'accuracy': 0.9747999999999999}, {'accuracy': 0.9745700000000002}, {'accuracy': 0.9748299999999999}, {'accuracy': 0.9742}, {'accuracy': 0.9738199999999999}, {'accuracy': 0.97409}, {'accuracy': 0.8949999999999999}, {'accuracy': 0.96595}, {'accuracy': 0.96936}, {'accuracy': 0.96722}, {'accuracy': 0.9692000000000001}, {'accuracy': 0.96926}, {'accuracy': 0.97033}, {'accuracy': 0.97021}, {'accuracy': 0.9697000000000001}, {'accuracy': 0.97058}, {'accuracy': 0.9712999999999999}, {'accuracy': 0.97129}, {'accuracy': 0.97179}, {'accuracy': 0.9723799999999999}, {'accuracy': 0.9714799999999999}, {'accuracy': 0.9715299999999999}, {'accuracy': 0.9714699999999998}, {'accuracy': 0.9728599999999998}, {'accuracy': 0.97223}, {'accuracy': 0.9724799999999998}, {'accuracy': 0.97197}, {'accuracy': 0.9729599999999999}, {'accuracy': 0.97387}, {'accuracy': 0.9732700000000001}, {'accuracy': 0.9739799999999998}, {'accuracy': 0.9728900000000001}, {'accuracy': 0.9745799999999999}, {'accuracy': 0.9734}, {'accuracy': 0.9730000000000001}, {'accuracy': 0.97423}, {'accuracy': 0.9743399999999998}, {'accuracy': 0.9726000000000001}, {'accuracy': 0.9744400000000001}, {'accuracy': 0.9732899999999999}, {'accuracy': 0.9753000000000001}, {'accuracy': 0.97388}, {'accuracy': 0.9723499999999999}, {'accuracy': 0.97478}, {'accuracy': 0.97547}, {'accuracy': 0.9745900000000001}, {'accuracy': 0.9753200000000002}, {'accuracy': 0.97428}, {'accuracy': 0.9752599999999999}, {'accuracy': 0.97438}, {'accuracy': 0.9746600000000001}, {'accuracy': 0.97495}, {'accuracy': 0.97625}, {'accuracy': 0.9747399999999999}, {'accuracy': 0.9761200000000001}, {'accuracy': 0.9740399999999999}, {'accuracy': 0.97187}, {'accuracy': 0.97513}, {'accuracy': 0.9763199999999997}, {'accuracy': 0.97417}, {'accuracy': 0.9752899999999999}, {'accuracy': 0.9748000000000001}, {'accuracy': 0.97568}, {'accuracy': 0.97631}, {'accuracy': 0.97424}, {'accuracy': 0.9754200000000001}, {'accuracy': 0.97654}, {'accuracy': 0.97691}, {'accuracy': 0.97427}, {'accuracy': 0.9766799999999998}, {'accuracy': 0.97711}, {'accuracy': 0.9767899999999999}, {'accuracy': 0.9760900000000001}, {'accuracy': 0.9721799999999998}, {'accuracy': 0.97606}, {'accuracy': 0.9747399999999999}, {'accuracy': 0.97488}, {'accuracy': 0.97528}, {'accuracy': 0.9743999999999998}, {'accuracy': 0.9748299999999999}, {'accuracy': 0.97503}, {'accuracy': 0.9758899999999999}, {'accuracy': 0.97449}, {'accuracy': 0.9764100000000001}, {'accuracy': 0.97704}, {'accuracy': 0.97675}, {'accuracy': 0.9763399999999999}, {'accuracy': 0.97607}, {'accuracy': 0.9772000000000001}, {'accuracy': 0.9747199999999999}, {'accuracy': 0.976}, {'accuracy': 0.9771000000000001}, {'accuracy': 0.97585}, {'accuracy': 0.97722}, {'accuracy': 0.97552}, {'accuracy': 0.97736}, {'accuracy': 0.9763399999999999}, {'accuracy': 0.9758699999999999}, {'accuracy': 0.97624}, {'accuracy': 0.9759300000000002}, {'accuracy': 0.9759800000000001}, {'accuracy': 0.9762700000000001}, {'accuracy': 0.9774800000000001}, {'accuracy': 0.97507}, {'accuracy': 0.97644}, {'accuracy': 0.9760099999999999}, {'accuracy': 0.97521}, {'accuracy': 0.9765}, {'accuracy': 0.9749599999999999}, {'accuracy': 0.97521}, {'accuracy': 0.97711}, {'accuracy': 0.97683}, {'accuracy': 0.9770300000000001}, {'accuracy': 0.9776300000000001}, {'accuracy': 0.9773700000000002}, {'accuracy': 0.97674}, {'accuracy': 0.97669}, {'accuracy': 0.9768399999999999}, {'accuracy': 0.9775499999999999}, {'accuracy': 0.97539}, {'accuracy': 0.9755400000000002}, {'accuracy': 0.97669}, {'accuracy': 0.9776}, {'accuracy': 0.9778500000000001}, {'accuracy': 0.9780600000000002}, {'accuracy': 0.97507}, {'accuracy': 0.97783}, {'accuracy': 0.97743}, {'accuracy': 0.97676}, {'accuracy': 0.97773}, {'accuracy': 0.9780099999999999}, {'accuracy': 0.9749199999999998}, {'accuracy': 0.9767700000000001}, {'accuracy': 0.9768100000000001}, {'accuracy': 0.9767999999999999}, {'accuracy': 0.97693}, {'accuracy': 0.97682}, {'accuracy': 0.9765099999999999}, {'accuracy': 0.9756699999999998}, {'accuracy': 0.9771500000000002}, {'accuracy': 0.9764800000000001}, {'accuracy': 0.97679}, {'accuracy': 0.97651}, {'accuracy': 0.97635}, {'accuracy': 0.97524}, {'accuracy': 0.97821}, {'accuracy': 0.9761999999999998}, {'accuracy': 0.97638}, {'accuracy': 0.97683}, {'accuracy': 0.9760099999999999}, {'accuracy': 0.9780099999999999}, {'accuracy': 0.9765600000000001}, {'accuracy': 0.9773}, {'accuracy': 0.9772400000000001}, {'accuracy': 0.9781299999999999}, {'accuracy': 0.97707}]\n"
     ]
    }
   ],
   "source": [
    "print([ev for _, ev in report.get_evaluation(False)])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7b1d25a1-d183-4e82-bbf4-24d4a64cda66",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "49eb0aaf-7df8-497b-86ac-24fb6d6bbc85",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4962adf7-d916-4138-9802-9148ab602638",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b9a01a27-105c-4a38-bcd6-27a03e934dbc",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fac72374-f0a5-4f97-9691-af5e04fc2467",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "badcd6ec-9721-42c9-9dd5-9b5130db3188",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5b6fe361-f901-46d7-9d2c-4115348e2c32",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0177a53a-3a26-4480-88bb-03886a575144",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a4e1917d-7c35-48eb-853d-65387d956fee",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "26ee4fad-c628-4bbd-95f5-b0df5ab294b0",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4b3627d2-b04b-4cfb-89a9-524a61639232",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f8fa9d07-be9f-4542-aa8e-5c30ff0e71e4",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4dd86bc0-0131-4841-9539-1b06169147a7",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d5e81a1f-84dc-4bca-b652-4d33b9dd8689",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b1866f4e-41d8-4b65-b56e-040692dd9b2f",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5ebed083-ab7d-4460-8ecb-dba0d947fbc4",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "afa328ba-a3df-4629-ac87-aba34ba8d4bf",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5206c7b8-1cde-42ba-93a6-7180536ee869",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fb186141-bfa8-46f0-8b20-71a75c896494",
   "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.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
