{
 "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[:60000]]\n",
    "        dataset_train.targets = [dataset_train.targets[i] for i in indices_train[:60000]]\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[:10000]]\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": "beed7358-e1ad-4c49-abdf-bd8a582431d6",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "train_data = torch.load('train_data.pt')\n",
    "train_targets = torch.load('train_targets.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "a433b581-7eea-443f-b527-2df6222284d9",
   "metadata": {},
   "outputs": [],
   "source": [
    "test_data = torch.load('test_data.pt')\n",
    "test_targets = torch.load('test_targets.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "67f02a07-6dc6-46c5-9262-430b2eba5b27",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "b4295861-8fc1-48a6-a220-9c9d8da69663",
   "metadata": {},
   "outputs": [],
   "source": [
    "def dirichlet_distribution(alpha, num_clients):\n",
    "    return np.random.dirichlet([alpha] * num_clients)\n",
    "\n",
    "def partition_dataset(data, targets, num_clients, alpha):\n",
    "    num_classes = 14\n",
    "    data_indices = [[] for _ in range(num_clients)]\n",
    "    labels = np.array(targets)\n",
    "    \n",
    "    for k in range(num_classes):\n",
    "        class_indices = np.where(labels == k)[0]\n",
    "        np.random.shuffle(class_indices)\n",
    "        proportions = dirichlet_distribution(alpha, num_clients)\n",
    "        proportions = (proportions / proportions.sum()) * len(class_indices)\n",
    "        proportions = np.cumsum(proportions).astype(int)\n",
    "        \n",
    "        previous_idx = 0\n",
    "        for i in range(num_clients):\n",
    "            client_indices = class_indices[previous_idx:proportions[i]]\n",
    "            data_indices[i].extend(client_indices)\n",
    "            previous_idx = proportions[i]\n",
    "    \n",
    "    return data_indices"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "a411b9b0-00c9-460e-80c2-1f21ac4085f7",
   "metadata": {},
   "outputs": [],
   "source": [
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "dataset = TensorDataset(test_data, test_targets)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "257703e7-5e07-494d-8afa-b8dcc949d2a4",
   "metadata": {},
   "outputs": [],
   "source": [
    "num_clients = 10\n",
    "alpha = 1.5  # Example alpha value\n",
    "data_indices = partition_dataset(train_data, train_targets, num_clients, alpha)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "0b1c88a4-e7bb-45aa-8559-6d56c4195090",
   "metadata": {},
   "outputs": [],
   "source": [
    "concatenated_data = []\n",
    "concatenated_targets = []\n",
    "for indices in data_indices:\n",
    "    concatenated_data.append(train_data[indices])\n",
    "    concatenated_targets.append(train_targets[indices])\n",
    "\n",
    "concatenated_data = torch.cat(concatenated_data, dim=0)\n",
    "concatenated_targets = torch.cat(concatenated_targets, dim=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "40e9e1b8-72d3-4450-b7b6-437d30cfe00b",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e3331da3-b4fd-4dfd-80cf-7aa99bdd710e",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "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(TorchModel):\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": null,
   "id": "7cabf73b-6227-4713-9e8a-f567959339f8",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "9e08faa4-e0d0-4358-b1c7-fb5a17410fb2",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "\n",
    "net = ResNet50(pretrained=False)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "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=None`.\n",
      "  warnings.warn(msg)\n"
     ]
    }
   ],
   "source": [
    "model = models.resnet50(pretrained=False)\n",
    "net.load_state_dict(model.state_dict())\n",
    "net.fc = nn.Linear(2048, 14)\n",
    "# netglob = netglob.to(args.device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "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": null,
   "id": "c1f84f4b-c630-4da5-82c3-211a0a11aa6b",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "659891a0-b4b7-4ef2-8cae-4215a601ce98",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d930ddae-ecf6-49f7-b449-bc5d3f14ce68",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b8277fc5-947f-4cc1-9ae9-8afcc8d27896",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "ec544b6d-bff3-428f-98d6-a34bd2218a8e",
   "metadata": {},
   "outputs": [],
   "source": [
    "Xtr, ytr = (concatenated_data, concatenated_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": 13,
   "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": null,
   "id": "b618a6b5-e2aa-489c-a761-915f75fd2151",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "d7593361-9cfa-4a05-b0ae-5f83fe29204f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "nodes = PENSNode.generate(\n",
    "    data_dispatcher=data_dispatcher,\n",
    "    p2p_net=StaticP2PNetwork(10),\n",
    "    model_proto=TorchModelHandler(\n",
    "        net=net,\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=200)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b7599863-d1a1-416e-9d03-3f4ef9c3d49e",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "e1348413-a05d-4797-a755-1ab12c8b91cf",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAm4AAAHWCAYAAADO2QWWAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy81sbWrAAAACXBIWXMAAA9hAAAPYQGoP6dpAACCUElEQVR4nO3dd3hT9f4H8HeSNunee9DNLhQKVKYIlcLlKiiy9AqCyM8BihVFrgJuEBd6QVCugusqbpwFrJS9oWxKKS3de6QzaZPz+yPNoaEtHbRN2r5fz5PnoScnJ5/TFPrmOyWCIAggIiIiIpMnNXYBRERERNQ8DG5EREREnQSDGxEREVEnweBGRERE1EkwuBERERF1EgxuRERERJ0EgxsRERFRJ8HgRkRERNRJMLgRERERdRIMbkTUZaWkpEAikWDr1q3isZdeegkSicR4RXUhW7duhUQiQUpKirFLIeo2GNyIiBrwv//9D+vWrWvX98jMzMRLL72E+Pj4dn2fjvThhx8aBGUialsMbkREDeio4Pbyyy8zuBFRszG4EVGnUVVVBa1Wa+wyTEp5ebmxSyCiDsTgRkRNOnXqFCZNmgQ7OzvY2Nhg/PjxOHz4sPj88ePHIZFI8Nlnn9V77Y4dOyCRSPDbb7+JxzIyMjB//ny4u7tDoVCgX79++PTTTw1eFxcXB4lEgm+++QYvvvgivL29YWVlBaVSicLCQixduhShoaGwsbGBnZ0dJk2ahNOnT7fJ/Y4dOxa///47rl27BolEAolEAn9/f/F5lUqFVatWITg4GAqFAr6+vnjuueegUqkMrrNr1y6MGjUKDg4OsLGxQa9evfDvf/9bvL+hQ4cCAObNmye+z81aq/Tj8y5cuID7778fjo6OGDVqlPj8l19+ifDwcFhaWsLJyQmzZs1CWlqawTUSExMxbdo0eHh4wMLCAj4+Ppg1axZKSkoANDwuUE8ikeCll15qtD5/f3+cP38ee/bsEe9n7NixAIDq6mq8/PLLCAkJgYWFBZydnTFq1Cjs2rWr0esRUX1mxi6AiEzb+fPnMXr0aNjZ2eG5556Dubk5PvroI4wdOxZ79uxBREQEhgwZgsDAQHz77beYO3euweu3bdsGR0dHREVFAQBycnJw2223QSKRYNGiRXB1dcWff/6Jhx9+GEqlEkuWLDF4/auvvgq5XI6lS5dCpVJBLpfjwoUL+PnnnzF9+nQEBAQgJycHH330EW6//XZcuHABXl5et3TPL7zwAkpKSpCeno733nsPAGBjYwMA0Gq1uPvuu7F//34sXLgQffr0wdmzZ/Hee+/h8uXL+Pnnn8Xv2z//+U8MGDAAr7zyChQKBa5cuYIDBw4AAPr06YNXXnkFK1euxMKFCzF69GgAwIgRI5qsb/r06QgJCcEbb7wBQRAAAK+//jpWrFiBGTNmYMGCBcjLy8N//vMfjBkzBqdOnYKDgwPUajWioqKgUqmwePFieHh4ICMjA7/99huKi4thb29/S9+3devWYfHixbCxscELL7wAAHB3dwegC52rV6/GggULMGzYMCiVShw/fhwnT57EnXfeeUvvS9StCERENzF16lRBLpcLSUlJ4rHMzEzB1tZWGDNmjHhs+fLlgrm5uVBYWCgeU6lUgoODgzB//nzx2MMPPyx4enoK+fn5Bu8za9Yswd7eXqioqBAEQRB2794tABACAwPFY3pVVVWCRqMxOJacnCwoFArhlVdeMTgGQNiyZYt4bNWqVUJz/umbPHmy4OfnV+/4F198IUilUmHfvn0Gxzdt2iQAEA4cOCAIgiC89957AgAhLy+v0fc4duxYvfpuRl/77NmzDY6npKQIMplMeP311w2Onz17VjAzMxOPnzp1SgAgfPfdd42+R0PfMz0AwqpVq8Svt2zZIgAQkpOTxWP9+vUTbr/99nqvHThwoDB58uSmb5KIbopdpUTUKI1Gg507d2Lq1KkIDAwUj3t6euL+++/H/v37oVQqAQAzZ85EdXU1fvzxR/G8nTt3ori4GDNnzgQACIKAH374AXfddRcEQUB+fr74iIqKQklJCU6ePGlQw9y5c2FpaWlwTKFQQCqVijUWFBSIXZE3vr6tfffdd+jTpw969+5tUP+4ceMAALt37wYAODg4AAC2b9/e5uPyHn30UYOvf/zxR2i1WsyYMcOgJg8PD4SEhIg16VvUduzYgYqKijatqSkODg44f/48EhMTO/R9iboaBjcialReXh4qKirQq1eves/16dMHWq1WHEM1cOBA9O7dG9u2bRPP2bZtG1xcXMRQk5eXh+LiYnz88cdwdXU1eMybNw8AkJuba/A+AQEB9d5bq9XivffeQ0hICBQKBVxcXODq6oozZ86IY7XaS2JiIs6fP1+v/p49exrUP3PmTIwcORILFiyAu7s7Zs2ahW+//bZNQtyN35PExEQIgoCQkJB6dV28eFGsKSAgANHR0fjvf/8LFxcXREVFYcOGDe3+PQOAV155BcXFxejZsydCQ0Px7LPP4syZM+3+vkRdDce4EVGbmTlzJl5//XXk5+fD1tYWv/zyC2bPng0zM90/NfrQ8q9//aveWDi9AQMGGHx9Y2sbALzxxhtYsWIF5s+fj1dffRVOTk6QSqVYsmRJu8861Wq1CA0Nxbvvvtvg876+vmLde/fuxe7du/H7778jJiYG27Ztw7hx47Bz507IZLJW13Dj90Sr1UIikeDPP/9s8Lr68XkA8M477+Chhx7C9u3bsXPnTjz55JNYvXo1Dh8+DB8fn0YXJ9ZoNK2uFwDGjBmDpKQk8X3/+9//4r333sOmTZuwYMGCW7o2UXfC4EZEjXJ1dYWVlRUSEhLqPXfp0iVIpVIxqAC64Pbyyy/jhx9+gLu7O5RKJWbNmmVwPVtbW2g0GkRGRra6ru+//x533HEHPvnkE4PjxcXFcHFxafV162oswAQFBeH06dMYP358kzswSKVSjB8/HuPHj8e7776LN954Ay+88AJ2796NyMjINtvBISgoCIIgICAgQGz5u5nQ0FCEhobixRdfxMGDBzFy5Ehs2rQJr732GhwdHQHovpd1Xbt2rVm13OyenJycMG/ePMybNw9lZWUYM2YMXnrpJQY3ohZgVykRNUomk2HChAnYvn27wbZGOTk5+N///odRo0bBzs5OPN6nTx+EhoZi27Zt2LZtGzw9PTFmzBiD602bNg0//PADzp07V+/98vLyml2XUDubUu+7775DRkZGC++wcdbW1g12Ic6YMQMZGRnYvHlzvecqKyvFddUKCwvrPR8WFgYA4rIh1tbWAOqHpJa69957IZPJ8PLLL9f7vgiCgIKCAgCAUqlETU2NwfOhoaGQSqViTXZ2dnBxccHevXsNzvvwww+bVYu1tXWD96OvQc/GxgbBwcH1llAhoptjixsR3dRrr70mrkf2+OOPw8zMDB999BFUKhXWrl1b7/yZM2di5cqVsLCwwMMPPyxOItBbs2YNdu/ejYiICDzyyCPo27cvCgsLcfLkSfz1118NBp4b/fOf/8Qrr7yCefPmYcSIETh79iy++uorgwkUtyo8PBzbtm1DdHQ0hg4dChsbG9x111148MEH8e233+LRRx/F7t27MXLkSGg0Gly6dAnffvstduzYgSFDhuCVV17B3r17MXnyZPj5+SE3NxcffvghfHx8xLXXgoKC4ODggE2bNsHW1hbW1taIiIhocFzfzQQFBeG1117D8uXLkZKSgqlTp8LW1hbJycn46aefsHDhQixduhR///03Fi1ahOnTp6Nnz56oqanBF198IQZqvQULFmDNmjVYsGABhgwZgr179+Ly5cvN/r5t3LgRr732GoKDg+Hm5oZx48ahb9++GDt2LMLDw+Hk5ITjx4/j+++/x6JFi1p0r0TdnhFntBJRJ3Hy5EkhKipKsLGxEaysrIQ77rhDOHjwYIPnJiYmCgAEAML+/fsbPCcnJ0d44oknBF9fX8Hc3Fzw8PAQxo8fL3z88cfiOfrlQBpauqKqqkp45plnBE9PT8HS0lIYOXKkcOjQIeH22283WIriVpYDKSsrE+6//37BwcFBAGCwNIharRbefPNNoV+/foJCoRAcHR2F8PBw4eWXXxZKSkoEQRCE2NhYYcqUKYKXl5cgl8sFLy8vYfbs2cLly5cN3mf79u1C3759BTMzsyaXBtHX3tgSIz/88IMwatQowdraWrC2thZ69+4tPPHEE0JCQoIgCIJw9epVYf78+UJQUJBgYWEhODk5CXfccYfw119/GVynoqJCePjhhwV7e3vB1tZWmDFjhpCbm9us5UCys7OFyZMnC7a2tgIA8fN47bXXhGHDhgkODg6CpaWl0Lt3b+H1118X1Gp1E58EEdUlEYQb2tWJiIiIyCRxjBsRERFRJ8HgRkRERNRJMLgRERERdRIMbkRERESdBIMbERERUSfB4EZERETUSXAB3gZotVpkZmbC1ta2zbakISIiImqMIAgoLS2Fl5dXvYXLbzzR6NavXy/4+fkJCoVCGDZsmHDkyJFmve7rr78WAAhTpkwxOK7VaoUVK1YIHh4egoWFhTB+/Ph6i17eTFpamriAKB988MEHH3zwwUdHPdLS0m6aUYze4qbfUmbTpk2IiIjAunXrEBUVhYSEBLi5uTX6upSUFCxduhSjR4+u99zatWvxwQcf4LPPPkNAQABWrFiBqKgoXLhwARYWFk3WZGtrCwBIS0sz2IeRiIiIqD0olUr4+vqKGaQxRt85ISIiAkOHDsX69esB6LopfX19sXjxYjz//PMNvkaj0WDMmDGYP38+9u3bh+LiYvz8888AAEEQ4OXlhWeeeQZLly4FAJSUlMDd3R1bt27FrFmzmqxJqVTC3t4eJSUlDG5ERETU7pqbPYw6OUGtVuPEiROIjIwUj0mlUkRGRuLQoUONvu6VV16Bm5sbHn744XrPJScnIzs72+Ca9vb2iIiIaPSaKpUKSqXS4EFERERkaowa3PLz86HRaODu7m5w3N3dHdnZ2Q2+Zv/+/fjkk0+wefPmBp/Xv64l11y9ejXs7e3Fh6+vb0tvhYiIiKjddarlQEpLS/Hggw9i8+bNcHFxabPrLl++HCUlJeIjLS2tza5NRERE1FaMOjnBxcUFMpkMOTk5BsdzcnLg4eFR7/ykpCSkpKTgrrvuEo9ptVoAgJmZGRISEsTX5eTkwNPT0+CaYWFhDdahUCigUChu9XaIiIiI2pVRW9zkcjnCw8MRGxsrHtNqtYiNjcXw4cPrnd+7d2+cPXsW8fHx4uPuu+/GHXfcgfj4ePj6+iIgIAAeHh4G11QqlThy5EiD1yQiIiLqLIy+HEh0dDTmzp2LIUOGYNiwYVi3bh3Ky8sxb948AMCcOXPg7e2N1atXw8LCAv379zd4vYODAwAYHF+yZAlee+01hISEiMuBeHl5YerUqR11W0RERERtzujBbebMmcjLy8PKlSuRnZ2NsLAwxMTEiJMLUlNTb76CcAOee+45lJeXY+HChSguLsaoUaMQExPTrDXciIiIiEyV0ddxM0Vcx42IiIg6UqdYx42IiIiImo/BjYiIiKiTYHAjIiIi6iQY3IiIiIg6CQY3IiIiok6CwY2IiKgNHbiSj4e2HEVmcaWxS6EuiMGNiIioDb362wXEJeThi8PXjF1Kt1JYrsabMZeQWlBh7FLaFYMbERFRG0nILsWl7FIAwJGrBUaupntZ9ct5bIxLwut/XBCPFZWr8WHcFWSVdJ3WT6PvnEBERNRVbI/PEP98Jr0EFeoaWMmv/6qtVGvw9LZ4DPC1x+Njg1v1HvllKnx1OBUV1TVYOqEXzGVsg0nILsVvZzIBAHsu56FSrYGlXIa3dybgqyOp+PpoKr79v+HwtLds8lqHkgqQlFcGN1sFbC3MUViuRm5pFQBg3siAdr2P5mBwIyIiagNarYDt8Zni1zVaASeuFWF0iKt4bOeFbMSc1z16udtifB/3Zl9fVaPB6j8u4eujqVDVaAEAfTzsMHWQd9vdRCf13q7L0O8DVVWtxZ7LeRjbyxW/ntZ9HmmFlXhg8xFs+7/hcLVVNHqdT/cn45XfLjT4nIuNwiSCG2M6ERFRGziRWoSM4krYKMzwj1APAMCRq4UG58RezBX/vOyHsygoUzX7+h/vuYqtB1OgqtHCyVoOAPj2eFqD53bV3Sx/PZ2J9/9KhEZ7/f7OZZQg5nw2JBJgbC9dSN55PhuxF3OhrKqBu50C3g6WuJpfjjvf24MH/nsYr/9+Afk3fO+/O54mhrZhAU4Y6OuAQBdrDPFzxD9CPXDXQE+T+L6yxY2IiKiVdpzPxtqYSxjX2w1ZJbrutKh+HhgW4Ig/zmbjSPL1cW41Gi3iEnTBzclajvwyFZb/eBbvzQyDteLmv45LKquxed9VAMDr9/TH7T1dMXrtbhxMKkBqQQV6OFuJ5+66kIPlP57F3OF+WDw+pK1vuUmF5Wos2RYPQRAQEeCEIf5O6ONpB3tL81u6bklFNZ757jTUNVp4OVhg+hBfAMA7OxMAAFMGeuH+CD/EJeThr4s5yC9XAwDuC/fB9HBfPPDfI8gorsSBKwU4cKUAheXVeGfGQADA7oRcLPvhDABgwagAvDC5DyQSyS3V214Y3IiIiFpBXaPFS7+cR1ZJFZLyksXjU8K80MNJF6Ti04rF8VYnrhVBWVUDRytzfDZvGO7deAA7L+Sg/0s7EOhijdt7umHmUF/08rBFSUU1ruaXIdjNBrYW5vh0fzKUVTXo6W6D2UN7QCqVYFSwC/Yl5uO7E2l4ZkIvAEBSXhme3haPMlUN3tl1GR721wOOIAhNhpE/zmbh5V/Pw0ZhBg97C/g5WyPU2x4+jpY4k16CE9eKEOJug+eiekMmlSCzuBJLvzsNP2crvHR3P5hLpXjm23jsvZwHANiXmC9e29PeAk/f2RMzautpSrmqBmWqGrjbWQAAfjmdAXVtF/H7sYmYEuaN3Qm52J2QB5lUgqcie6KHkxVcbOTIL1OLNdwzyAf+LtaIfeZ2XMouxfGUQrz2+0X8cjoDz0b1gpO1HCu3n4NWAKaH+5h0aAMY3IiIiOoRBAHnM5Xo62kHqVT3S/zEtSK8t+synr6zJ8L9HPHTqXRklVTB1VaBEDcbHEwqgL+zFUYEOUMmlcDDzgLZyiqcSi3CiGAX/H1J19p2Ry83hPrY481pA/DWjoTa4FeOpLxkfHogGS42CrEbz9lajucm9sKn+3XBcElkT7GemUN9sS8xH9+fSMeSyJ5Q1Wjw2JcnUKaqgbO1HAXlavz7p7MoV9XgYFIB9ibm4b5wH7zwj76wlMvq3XN2SRWW/XAGpVU1yIEKSXnlOHCl/szYvy/lIr2wEs9G9cKcT48itbACB5MKkF5UiXA/R+xOyIPCTIonx4fgQqYSp1KLkFlShaySKrzw01mMCHKGj6NVvevWVVWtwbSNB3E1vxzfPzocA3wc8N2JdPH59KJKbN53FZ8fSgEALBwTiAAXawDAnX3d8fVRXRfyQF8HBLvZAAAszGUI83VAmK8Ddl7IwdHkQmw5kAxvR0ukFVbCxUaBl6f0M+nQBjC4ERER1bNh9xW8vfMyFo8LFluz3vjjIk5cK8KFLCV+fnwkNsYlAQD+b0wgFowORGpBBWwszGBWO8szItAJ2+MzcTi5ECOCXRBbG9zG9XEDANw72Af3DvZBXqkKJ1OL8NPJDF0XX21os1WYoaBcjWU/nAUA9PawxcR+HmKNd/Z1h6OVObJKqvD8D2eQkFOKyzllcLVV4LfFo/DSL+fx57lsvPTr9cH2Xx5OxZGrhfhg9iD08bQTjwuCgOU/6kLbQB97LJvYG9nKKlzOKcPZjGKkF1Wir6cdAl2tsXlvMn4/m4Ud57NRoxXg7WCJwnI19iXmiy1sL9/dD7OG9RCvr6yqxsLPj+Pw1UKs//sK1kwbcNPv/9s7EsRlVVb8fA5rpg3AmfQSmEklWDQuGOv+SsRbO3RdpP7OVniqTpfwhH4eYnCbNrjhiRv/NyYQR5ML8b8jqVCY60LsU+ODDWYAmyrTr5CIiKiO3NIqPP7lSdw10AtzR/i3+jo1Gi0EoN5yGhXqGvy3toVry4EUPDImEGmFFThxrQiAbgzXlA37UVRRDQcrc8yuDSh1x5kBQESAM7bHZ2Ln+WyMDnHBldwymEklBrNMAcDVVoGofh6I6ueB/DIVrhWUI9jVFpZyGT6Mu4INu6+gWiMg+s7rrW0AoDCTYeogb2w5kCK2RpnLJFg/exDc7SzwzoyBKChX42peOaaGeaG/tz1e/+MiEnPLcO+HB/HlgmEI93MCAHx/Ih27E/Igl0nx9vSBCHG3bfT7FhHgjIVfHEdVtRb+zlb4euFtyCyuxENbjqG0qgZTw7wwc6hhd6idhTmejeqFaRsP4bsT6Xj09iD417aQ3ejI1QJ8ciC59h6lOJ1egse+PAEAGN/HDY+PDcZ3x9ORUbszxRv3hsLC/HoL4oggZ3jZW6BcrcE/B3g1+B539HJDsJsNruSWoVRVgx5OVpg5tEeD55oaBjciImoX6UUV+PlUBmYM8YVb7TiltrDtaBqOXytCUl4Z/nWbH2TSlndtabUCpn90CNklVYhZMsZg4Pz3J9JRXFENAChT1eDrI6lIKSgHoAsFCdmlKKgd+D5/ZECjEwvG9HSBwkyKS9mlmL7pEABgqL/TTQfpu9go4GJzfbmKJZE9MTXMG9nKKtwW6Fzv/EdvD0JqQQXsrcwxyNcBo0NcxUBkJTfDt/833OD80SEuWPz1KRxMKsBDW47h60duw+GrBWLr1dN39rxpaNPdlyu+/b/h+ONsNuaN9Ie7nQU87S3x8xMjcTCpANPDfRrsbgz3c8LtPV2x53IePohNxLszw8TnBEFAbqkKx1OKsCbmIgQBmDnEF709bfHyrxeQUrsbwvRwX8jNpHh+Um8s/voU5gz3w4ggF4P3UZjJ8MviUdBoBXH27Y2kUgkeGR0gtmZG39kTcrPOsdCGRDCFua0mRqlUwt7eHiUlJbCzs2v6BUREVE/0tnj8eCoDHnYW+O/cIejvbW/wfHZJFQ5cycfp9GKUqWqw6p/9YG/V9MzDSe/vw8UsJQDgh8dGINzPscW1HUspFMPUO9MHYlq4DwBAoxVwx9txSC2swFB/RxxLKYKbrQKlVTWorNZg28LbIJFIcP/mw7BWmGHvs3fctObjKYVY/eclsbXuxcl9sGB0YIvrbUuVag3mfHoEx1KKIJUA+pU1xvd2w0cPhotdve3hTHox7l5/oMnzvB0sEbNkNCzNZbh7/QFcyFLCxUaBQ8vHiS2kuaVVcLVRtHpMmqpGg/lbj8FaboZN/wo3aM00huZmD7a4ERFRuziaolvDLFtZhfs2HcS6mYMwsb9ujNblnFLcs+EAytUa8fw+HnZ4ZMzNQ01KfrkY2gBg96VcMbgl5pTC29GyWeOUfjp1fYeDmPPZYnDbeT4bqYUVcLQyx3/nDMWd7+1BbqluzFkvd1sMC3CCRCJBzJIxUJhJmwyaQ/yd8P2jw7Hnch4uZpXiweF+TdbW3izlMnzy0FDM/vgwzmcqYaswwwuT+2DmUN92H5g/wMcB9wzyNvj+60klQG8POwzxd8TDowJga6H73r45bQCe/OYUHh4VYNCt7WZ7a624CjMZvlpw2y1dwxgY3IiIqM3lllYhvagSEgkwMsgF+6/k48mvT+GnJ0agr6cdVm4/h3K1BoEu1nC2keNYShFOpRU1ed0/z2UD0I19UtVo8felXCyN6oU/zmbh8a90497+M3vQTa+hqtHg9zNZ4td7L+ehXFUDK7kMH+3VrZX24G1+sLcyx7yRAXgz5hIA4F/D/cRgo5+p2BwSiQRje7lhbC+3Zr+mvdlZmON/C27Db2czMa63W7O2gmor784YiH//ow8ECJBAAokEkEDXtdvQbNdQH3vsXjq2w+ozdZ2jQ5eIiDqVk9eKAehaqbbOG4rIPm5Qa7R48utT+O54Og5fLYTCTIrP5g/D03f2BADEpxY3ed0/z+kC1+JxwZBIgAtZSqQXVYhjtHacy0aZquam14hLyENJZTXc7RTwc7aCqkaL3Qm52HE+G/FpxbAwl2JO7aSH+yN6wMVGATdbBe7pYltL2VuZ44EIvw4NbYAuyLraKuBmawFXW92YPmcbRYOhjepjcCMiojZ3KlXXejaohyPMZFKsvW8g3O0USMorx3O1K9Q/cUcwfJ2sMMDHARIJkFlShVxlVaPXTCuswJn0EkglwKxhPTDQxwEA8My3p5Gcr5s8oK6zO0Fj9BvB3z3QS+y63R6fiTf+0LWsLRwdKE4QsLc0x66nx2DHkjGwaWJ3A6KOwOBGRERtTj8Yf3APBwC6LZ7enREG/RAqP2crLKwdz2ajMENPN91MxlNpxY1eM6a2m3RYgBNcbBQY11vX9XgkWTeWzqN25qr+vLp2ns/GS7+cx9YDyfirdr/QqYO8Mam/JwDdNlGphRVws1Xg/24PMnito7Ucjo3MTiTqaAxuRETUptQ1WpzJKAEADK4z43NksAuWTugFa7kMb9xjuPbWoNqAF3+T4PbbWV036T9CdWFLH9wA3TIa787U7TsZl5AHVc31SQ+lVdV48ptT2HowBS/9egHqGi16utugr6cdBvrYw8v++iD3Z6N6NblvKJExMbgREVGbupClhLpGCwcrcwTesMjqE3cE4/wrEzEy2HDtrTBfBwCNj3O7kluK02nFkEklYitZPy87uNvpujQfHxuE2wKc4W6nQJmqBgfrbNW043wOqqq1cLdTYHxvN/TxtMMzE3pBIpFAIpFgYu31+nvbYdpgn7b4FhC1G/63goiI2tRJsZvUsdnLS4TVtridSS+GRivgqyPX8NOpDLw/cxB6OFvhu+O6nQHu6OUGV1tdWJNIJHhvZhhOXivCg8P9IJVKMKGvB744fA0x57JxR22L3M+1S0/8K8IPi+tsjaS3eFwwzM0kuH9YD6Ov5UXUFLa4ERFRmzqRaji+rTlC3GxhLZehXK3BDyfS8fKvF3AqtRgv/HwWNRotfqwNX9OHGLaIjQhywaJxIeL6XvrJBn9dzIFGKyBHWYWDSbr9M6eENTwr1NFajuWT+sDPueEtmIhMCVvciIioTZ2q0+LWXDKpBAN8HHDoagGW/3QWmtrl/Pcl5mP5j2eRV6qCs7XcYFxbQ4YFOMHByhwF5Wqs++sy7C3NoRWAIX6O9fYSJeqM2OJGRERt5kpuGTJLqiCVAANrx601l767VKMV4GqrwEO1a6npN1CfOsi73obwNzKXSfHvSX0AAP/5+wo+iE0EAEzpYmuwUffF4EZERG3m66OpAHQzPls6OzOsTtBbO20Anp/UG75O1xeHvS+8eRMHZgz1xZJI3Vg2ZVUNzKQS/LN2JipRZ8fgRtRNVVVrIAiCscugLqSqWoPva1vHHoho+Z6ct/d0xYS+7ng2qhfu6O0GC3MZXrm7PyQSXVdnH8/GN96+0VPjQzB7mC8A4M6+7lyHjboMicB/uetRKpWwt7dHSUkJ7Oya/w8FUWexPT4Dz/9wFhP7e+C9mWHGLoe6iB9OpOOZ707D28ESe5+7A7I2mqGZmFMKN1uLJjd0v5FGK+DAlXwM9HFo8WuJOlpzswdb3Ii6EUEQsO6vy3jqm3hUVus22q5Q33xfRyIAKCpX440/LuKtHZcaPeerI9cA6Pb3bKvQBgAh7ratCl4yqQRjeroytFGXwlmlRN3Ie38lioO15WZSqGu0OJJciDt63XymHnVfgiDgm2NpWBtzCUUV1QCAOcP94W5nYXDehUwlTqYWw0wqqbdkBxG1Hba4EXUjP9SOP3pxch9MG6ybZbf3cp4xSyIT98vpTCz/8awY2gAgu6T+RvA/ndL9bEX184CbrUW954mobTC4EXUTucoqZBRXQiIBZg71xZgQVwC6dbKIGrO/9udj2mAfhHrbAwCylfWDm35T+ci+bL0lak8MbkTdxMnaPSB7udvC1sIcI4JcIJXUrrtVXGnc4shkJeaWAQDG93GDl4OuJS33huCmrtHiXKYSABDm2/xFd4mo5RjciLqJU7XbEA2qXc3e3spcXDdrXyK7S6k+QRBwpTa4hbjZiOPabmxxu5R9fVN5f+5OQNSuGNyITMT2+AyEvrQD246ltsv1Tzawf+To2u7SvZfZXUr1ZZZUoUxVA3OZBP4u1mJwy1GqDM47nVYMABjo49DsTeWJqHUY3IhMQFphBf7941mUVtXgxZ/PIb72F2FdGq3Q6gVz1TVanEkvAQAM9rvelTWmpy647b+SL+4NSaSXmFMKAAhwsYa5TFonuBm2uJ2q/XkNa+EWV0TUcgxuREam1QpY+t1plKs1kMukqNYIeOKrkygqV4vn5JWqMP6dOMz46FCD4e1CphKv/nahwdl+AHAxSwlVbVdWoIu1eHygjz1sLcxQUlmNiev24qVfzuNaQXmD1zibXoJtx1K57ls3kpij7ya1BQB4NBLc9P/RCKvTmktE7YPBjcjIth5MwZHkQljJZfjpiREIcLFGRnElFn99ClXVGjHYpRRU4FhKEdKL6k8keDPmEj7Zn4z7Nx9Gbmn98KbvJh3ka9iVZSaTiht5J+aWYevBFMz86LBBaASAkspqPPDfw1j2w1mMfSsO/zuSihqNtg2/C2SKLte2uIW42wAA3O0UAAyXAympqMbVPF3YH+jj0LEFEnVDDG5ERnQpW4k3Y3Qr0b8wuQ/6ednjwwcGw8Jciv1X8nH/5sN4PzYRe+qstabv8tSr0WhxPKUQAHA1vxwP/vcoCm8IXvoZpYN71J/x98yEXji54k58+MBgBLhYI1tZhX//dNagZe+TfVehrNK1tOWWqvDvn87ihZ/O3fo3gNpFQZmqTYJ1Yq5hi5u7va7FTVlVg0q1BgBwOr0YAODnbAUn7gdK1O4Y3IiMpFKtwaL/nYKqRos7erni/mE9AAB9PO2wdd4w2FmY4WRqMd6v3enAs/aX5pnaX5R6F7KUKFdrYKswg7udAgk5pbh97W7M33oMH+9Nwum0YpysXWOr7vi2upys5fhHqCc+mDUI5jIJ/jyXje+O6xZULSxX45P9yQCAD2YPwouT+wAAth1PE1vyyHQcSynE8NV/48739oqTBlqj7ozSnrUtbrYKM1iaywBc7y6N5/g2og7F4EZkJK/8dh5XcsvgZqvA29MHGnRh3hbojB8fHwEfR0sAQGQfNyyJDAGAehMXjibrWtuGBTjhqwW3wdvBEqWqGvx9KRdv/HEJUzYcEBfeHeBjf9OaQn3s8cyEXgCAVb+cx0d7kvBBbCLK1Rr087LDXQM8sWB0IO4L121p9MqvF6DlpAaT8vaOBKg1WiTnl2PaxoPYGJfUquvoZ5SaSXUzSgFAIpHAw95wnBuDG1HHYnAjMoK4hFx8fTQNEgnw3swwONso6p0T7GaLXxaNwrqZYfhg9iBxYdNzGSUGM0CP1AluwW422PvcHfht8Si8OLkPIvu4wdZCtyXx8EBn2Fo0vdn2wtGBuL2nKyqrNVj95yVsPZgCAFg6oZcYLp+L6gVruQzxacXYfjrjlr4XdGvSiypQVa3rtjxytQBHkgshl0lxZ1931GgFvBlzCeczS5q4Sn03zijVc7OtHeemrIIgCGKrHoMbUcdgcCMygrgE3Zi1GeG+GBns0uh5TtZyTB3kDSu5GYLdbGAll6FcrcHVPF0XllYr4FjK9eAGADKpBP297bFgdCD+O3co4ldOwF/Rt2PznCHNqk0qleCTuUPw1n0DxBa/IX6OGNvLVTzHzc4Cj98RDAB4888ElKk409QYruSWYfTa3Yh8dw8u55TiP39fAQBMH+KDjx8Mx8DaMJVaUNHia+tnlPZ0tzU4rm9xy1WqkFZYiYJyNeQyKfp62d3CnRBRczG4ERlBcr5uFl5Llk/QBzLgevdUYm4ZiiuqYSWXic819LpgNxtYK8ya/V5mMimmD/HF38+MxWfzh+GTuUPrLaz68KgA+DpZIltZhdd/v9DsaxeWq1u9Hh0Zik8rhiAA6UWVmLL+APZfyYeZVILHxgZBIpHAtbYlt+4G8c2ln1Ea7GZjcLzu7gknUnX/aejvbQeFmexWboWImonBjcgIrubrWjPqrqnWHANrx6jpZ5YeSS4AAIT7ORp0Z7UVuZkUt/d0hb1V/S5WC3MZ1k4bCAD4+mgadl/KRWJOKe798AAmvb8PxRXqeq/58vA1DH51F+5avx87z2czwEG3sPLT2+Lxxh8XW/z90K+5ZyaVoLK2u/S+cB/4OOq2nXKy1n1uRQ18Fk1JzG24xU3fVZqjrBI3lm9otjIRtY/m/xeciNqEqkYjrsUW4Nqy4Dagdp0s/cxScXybv1Ob1dcSw4Oc8fCoAHyyPxnR38ajQq2Bqka3DMWaPy9hzbQB4rlV1Rqs+0s3Q/ZchhILvziB2wKd8Pn8CMjNmg6dVdUaXMhS1luLrrO7mKXET6d04wT9nK3wQIRfs197rbYL9KnxISgoV+NMejGeHB8iPu9opVue48blYZpyJbcUF7J0m8brZ5Tq1Z2coF+/rbHZykTU9tjiRtTBUgsqIAiAjcJM7MpqLv0A8AtZSlSqNQYzSo3l2aheCHazQVFFNVQ1WnEv1G+OpYn1AcC2Y2nIL1PB28ESj48NgpVchsNXC/Hx3qZnPQqCgEX/O4V7PzyIN2MS2utWjCKpdrwiALz62wVxCY6GlFRUo7rO+mzXCnXBLdjNBi/d3Q8/Pj4SXg6W4vOOteuqtaTFrVxVg0e/PAl1jRYjgpwb7SpNzq/ApWxduAtncCPqMCYR3DZs2AB/f39YWFggIiICR48ebfTcH3/8EUOGDIGDgwOsra0RFhaGL774wuCchx56CBKJxOAxceLE9r4NomZJqm2lCHS1bnHLkY+jJRytzFGtETD27d3IK1VBYSYVB6Ebg4W5DOvvH4SRwc5YdVdffP/oCMwe5gsA+PdPZ6Gu0UJdo8WmPbqA9ujYIDw3sTfeuCcUAPDB31fEyRaN+e1MFv66mAMA2LQnCb+dyWzHO+pY+lYrAKiq1uKpb05BXVN/8dxzGSUYviYWCz8/Lh5Lre0q9XNuuOXWsbaL+8adMBojCAKe//EsruSWwd1OgfdnDar3M6rf9iq/TAWtAHg7WIphjojan9GD27Zt2xAdHY1Vq1bh5MmTGDhwIKKiopCbm9vg+U5OTnjhhRdw6NAhnDlzBvPmzcO8efOwY8cOg/MmTpyIrKws8fH11193xO0QNUk/MSGghePbAN06WvqQlqNUwdHKHG9NHwgLc+MODO/tYYevFtyGeSMDIJVKsGxib7jYyHEltwyzNx/GK7+dR1ZJFdxsFZheuwbclDAvjOnpCnWNFv/+6SxSCypwMrUIaYWGMyCLK9R4+dfzAICg2q7lZ787gwuZyjarP6ukssGw1BGu1v48PDwqAI5W5jifqcTXR1MNzqnRaPH8j2dQodZgX2I+VDUaKKuqxUkHPZytGry2vqu0uZMTdpzPwa+nM2EmlWDD/YPhalu/RfjGY4O4PylRhzL6GLd3330XjzzyCObNmwcA2LRpE37//Xd8+umneP755+udP3bsWIOvn3rqKXz22WfYv38/oqKixOMKhQIeHh7tWjtRa+hblwJdbJo4s2EPjwpAYbkaE/q646GRAbBpwWzRjuJgJcfr94Tiia9O4sS1InEQ+yOjA8WQKZFI8PrU/pjw3l4cvlqIMW/tFl/fx9MO43q7wt7SHIeSCpBfpkaImw1+WTQKC784jn2J+Xjk8+P4/rHh8LS3bLCG5jpwJR8PfnIE9wzywTszBt7StVpD//NwW6AzvB0s8cpvF/BzfAbm1u4hC+j2sz2XoQuqNVpBXKoDAFxs5I3+DLS0q1Q/2WX2sB4Y0si4SQtzGRytzMUwyG5Soo5l1BY3tVqNEydOIDIyUjwmlUoRGRmJQ4cONfl6QRAQGxuLhIQEjBkzxuC5uLg4uLm5oVevXnjsscdQUFDQ6HVUKhWUSqXBg6g1ErJLMXx1LL46cq3Rc8QWtxZOTNAbHeKKXxaNwqJxISYZ2vSi+nlg37I7sHhcMFxs5Ah0tcb9ET0MzvF1ssK/J/eBRAIozKTwsreATCrBxSwlNuxOwht/XMLuhDxIJMCaaQNgKZfhP7MHIdDFGhnFlZjzyVEUlauRmFOK1X9cxK4LOTetae/lPES+uwcbdl8Rj/1331VoBWB7fAbyy1Tt8r1ojCAIBi2w/xzoCakEOJVaLK69ll5UgXd3XQYAWMl1ofd8ZglSartJezg13NoG1Glxa2ZXqX4JkNAmdtio2zXKGaVEHcuo/+rn5+dDo9HA3d3d4Li7uzsuXbrU6OtKSkrg7e0NlUoFmUyGDz/8EHfeeaf4/MSJE3HvvfciICAASUlJ+Pe//41Jkybh0KFDkMnqdymtXr0aL7/8ctvdGHVbf57LQlZJFX4/k9Xo7ED9L+qWLgXSGXnaW+KZCb3wzIReEAShwTF9D97mh1lDfWEm1Y1HLSpX46+LOTieUgS1RguNVsCoEBexZcfBSo7PHx6G6ZsOITG3DHe+twf5ZbpgsuVgCnYuGSNu0VTXZwdT8MpvF6DRCnh312VMDvWETCpB3GXdYsg1WgE/nczAI2MC2/E7YihHqUKFWgOZVIIeTlaQm0kxMtgF+xLz8cvpDDxxRzBWbj+PCrUGw/ydEOpjj0/2J+N8plIMT42NbwOuj3FTVtWgRqOFWRNLxiRk64JbrxuWALmRu50FLmWXwsKcC+8SdTTT/e/6Tdja2iI+Ph5lZWWIjY1FdHQ0AgMDxW7UWbNmieeGhoZiwIABCAoKQlxcHMaPH1/vesuXL0d0dLT4tVKphK+vb7vfB3U9F2uXUMgtbbjlpqSiGgW1rR+tGePWmd1sIkbdNegcreWYPsQX04c0/nfQx9EKX9SGt/wyNSQSwNVGgdxSFVZsP4fP5w8T3y9XWYU3/riIn+N1ExrsLc1RUlmND/5OhKe9BQQBsDSXobJag2+Pp2HB6IAOW25E302qD20AcPdAL+xLzMfP8ZkIcLHB35dyYS6T4I17++Nshm79vvOZSqiqdWPy/BoZ3wbo7lUiAQQBKK6shstNZjHnl6nE72WI+8278d3tdNcZ4O3QLusHElHjjBrcXFxcIJPJkJNj2L2Rk5Nz0/FpUqkUwcG67XbCwsJw8eJFrF69ut74N73AwEC4uLjgypUrDQY3hUIBhaJlyzIQZZdUYd1fl3H3QC+MqN226mKWrsVCvwH3jfQL77rbKVq0kwHVF+xmi+8eHY6dF3IwOdQTWgGIem8v9iXm44+z2Rjq74htx9KwaU8SytUaSCTAsom9MTzQGVM2HMDPpzLEvVtfvrsfVv5yDom5ZYhPK8agDur+S2qg9TWqvwde+PkcruSW4fkfzgAAHh8bjGA3W+i3qL2YpYRMqguXNwtuZjIp7Cx0QbWoXH3T4KbvJu3hZAUr+c1/Nnt76FrZxvRsfLs2ImofRv3NIZfLER4ejtjYWEydOhUAoNVqERsbi0WLFjX7OlqtFipV42NT0tPTUVBQAE9Pz1stmQgAkFFcidkfH0ZqYQXOZpTg9ydHo7SqGqm1MyJLq2pQVa2pN9vzejdp6yYmkKFgN1sEu13v1nt0bBA+iE3E0u9OQ1WjEYPOoB4OWPnPvmIgi+zjhr8u5qKkshqutgrcM9gbh68W4MdTGdhyIAUHruTjh5MZKK5QQ12jhb+LNb5ZeJsY9NqKOFGlznhHOwtzjOvlhpjz2ShV1SDQ1RqP3xGkO8/FGgozKSrUGpxK1U346OF085ZbJ2u5Lrg1MbP0cm036Y07JTRkznA/hPVwQH+vm4+FI6K2Z/Q27ujoaGzevBmfffYZLl68iMceewzl5eXiLNM5c+Zg+fLl4vmrV6/Grl27cPXqVVy8eBHvvPMOvvjiC/zrX/8CAJSVleHZZ5/F4cOHkZKSgtjYWEyZMgXBwcEGs06JmmN7fAYWfn7cYBP1tMIKzPzokBjSLmYpoayqFscH6eUq6/9nQr9mV2snJtDNPT42CH7OVqis1oW2wT0csG5mGH54dIRBK9qSyJ7in2cN9YV57d6sAPDL6Uy8vfMykvPLUVRRjXK1Buczlfj2eHq999sen4ERq2OxPT6jVfVeFdf0MwzyU8K8xD+/cU+ouA+omUyK3p661q5qjS6V3qzFDQAcase5NbV7QkJO88a36esY3MOxWTteEFHbMnpfzcyZM5GXl4eVK1ciOzsbYWFhiImJEScspKamQiq9/o9DeXk5Hn/8caSnp8PS0hK9e/fGl19+iZkzZwIAZDIZzpw5g88++wzFxcXw8vLChAkT8Oqrr7I7lFrsoz1XcSFLidiLOZgS5g0AeO77M0gvqoS/sxWqqrW6zbavFSH9hvXHckqr6q2v1Z0mJhiDhbkMX8yPwL4reRgT4grfRmZc9ve2x/yRAdibmIcHb9NNIokIcEIvd1sk5JSij6cdFo4JQH8ve/x1MRdvxlzCZwdT8NAIf7GL8uujqfj3T2chCMCbf17CP0I9Wzzeq7E9ayP7umP2sB4IcrXGbYHOBs/187LD6bRiALrdN5xrl/xojFPtzNKG9o6tS/8fj54eTQc3IjIeowc3AFi0aFGjXaNxcXEGX7/22mt47bXXGr2WpaVlvcV4iVpL30qhbxnRagWcStN1UW16MByf7EvGdyfScTS5EMU3dEU11OKW1EDXGLWtHs5WeMC56f0+V97V1+BrqVSCrx6JQHZJFfp52YkTFLwdLbEx7gpSCysQl5CL8X3cseVAMl7+9QIAQCaVILN2JvHUQd5Nvm9V7WbwAMQ9a29scTOXSbH63tAGX9+vzizOHk5WTU6kcNDvV3qT4CYIAi7Xrg3XnBY3IjIetnNTtyQIAkoqbz7mRxAEceFSfeDKKK5EVbUWcpkUwa424h6hR5MLxRml+u6jGycoxKcVi/tQBrvyl6MpcrFRoL+3vUEYspKbYfYw3fpzWw+m4MO4K2JoWzgmEE9H6jZ1/2jvVQiCcNPrF5WrMWbtbkS+uwcHk/IhCICthRlcbG7ealZXvzrjyprqJgUAJ2tdV+mN/7GoK7OkCmWqGpjLJN1utjNRZ8PgRt3SmphLGPTKTpy4VtjoOZXVGqhqt0HS7y+qD14BLtYwk0kREaDrxjqTXixuuK3v2qq7JEhhuRpPfHUSNVoBE/t5NLpFEZmmf93mB6kE2JeYj7W1m9w/OT4Eyyf1xoO3+cNKLsPFLCX2Jebf9Dqb9iYht1SF9KJKPPrlSQC61raWLD/S28NW7K5tzs+R2OJ2kzFuCbU/u4EuNhy3RmTi+DeUuqV9l/OhFYAjyY0Ht7qz8K7mlUGrFa63mLnpurZ8nSzhYWeBao2AqmotLMyliKhthcutbXHTaAU89c0pZBRXIsDFGmunD2iv26J24utkhTv7Xl8ofNnE3oi+syckEgnsrcwxc6huYsPHe682eo1cZRU+O5gCQDdhQL83alALW7gszGXinq1+TcwoBXSzSoHrY9wOXsnHJ/uTDVoHE7J1P9cc30Zk+hjcqNvRaq9vM6QfY9SQutsEqWq0yCiuFINbUG1wk0gkYncpoBsf5GmvW9Fe3+L2y+kM7EvMh4W5FBv/NRh2bbykBHWMp+/siX5ednj9nv54bGyQwXMPjwoAAOy/kt9oy9b63VdQVa3FoB4O+HXRKHjV/pz09mx5WFo4JghD/BwR2cetyXMd68wqFQQBS7bF49XfLuBonf+0XBZnlHKZGiJTZxKTE4g6UpayCpW1A8TTbpgJWteNG3Mn5ZXhSu1YtxC367/ghgY44ZfTulX5+3jawc1W9wtZP8ZNv8H6g7f5iQuXUufT28MOvz85usHnfByt4GKjQH6ZCpnFlWIrl15aYQW+PpoKAHh2Qi/4Olnh+8dGYMf57JvuENGY+8J9cF+4T7POdRRnlVYjR6kS/0NxKq0YEbXd+gktWMONiIyLLW7U7egXPQWaaHG7YTB3Ul55va5SAGLXKKAbf6TfDkj/C1L/S7EfFyvt0vQtrdkl9XfN+CA2EdUaASOCnMVdNrwcLDFvZABs2nkHDUfr67NKz9VumQVAXFKkQl0jtrj18eR/LIhMHYMbdTtJudeDW0ZRJbTahmcCFt3Q5XXkagFKKqshlRjuMxrsaiOupRXqYy+2uJVUVqOqWoNL+o27OX6oS9Nv+p59w2zipLwy/HBSt3jv0qheHV6XvsWtpLIaZ9KLxeNn0nUh7uS1YtRoBXjZW8DH0bLD6yOilmFXKXU7V2vHtwGAWqNFbqkKHrWtJXXpu0ptFGYoU9Vgb2IeAN1A9bpbWUmlEvxn9iBczinF4NrV+RVmUqhqtDidVozSqhrIpBKu3dbFNdbi9u6uy9AKum22BnfQHqh16XdOEATgQFKBeDyjuBL5ZSocSdYdiwh0btHsViIyDra4UbeTVKerFADSixoe56ZvcRvUwwEAUFWtmwUY7Fp/APeIYBc8NDIAEokEEokEbrXdpfrlIXR7TMrqvY66Dn34r9vidj6zBL+fyQIARN/Z8a1tgG4xX1sL3f/R42u7R81qlxM5k16MI1d1kxTqdvkTkelicKNuR78Lgn5sUVptcLuSW4YtB5Khqe061Y9xC/czbCWpO76tMe613aX7alvp2E3a9XnY1W9xe3fnZQDAXQO90NfLeOPH9N2l+p/t8bWzUY8kF4phLuKGrbWIyDQxuFG3Uq6qQVbtL9YRQbpfVGmFugkKz/9wBi//egG7LmQDuN5V6utoJU44AK4vBXIz+ha3M7WDwXszuHV5N7a4FVeoEXspFwDE3RWMxbHOLNdAF2uMrJ0g8e2xNKg1WrjZKuDPRaGJOgUGN+pW9Ou3OVvL0d9bN8szvagCVdUanK4duK3fJUEf3Jys5Qiq0z3anBY3/QQF/RqnvbgMSJfnfkOLm75L3tPeot5epB1Nv5YbAPTztscAHwcA11uVhwU4cXwbUSfB4EZd3p7LeXjiq5PIKK4Uf5kGudqIM+jSCitxNqME1Rpdysoo1rXAFZXrfqk5WJkbTCxoVnCr00IHcOPu7kDf4lamqkGZqkb8D4ApTEpxsrre4tbfyw59PG1hLrse1NhNStR5cFYpdWkHk/LxyGfHodZooRUEhNQGqEBXa/g66bqG0osrcDylSHxNRu3abvoWN0er6y1u7naKZu18oB/jBgBWchmXWegGbBRmsFWYoVRVg+ySKnEsZZCRW9uA6/uVArr1BBVmMvTxtBOXBLmNExOIOg22uFGntS8xD+PeiRN3JrjRuYwSLPz8BNQa3WzQP89l44+zuhl+Qa428HXUBbfM4iocTTZcJqGqWoMKtW53BUdrubit1W3NbJmo2+LW090WUim7oboDjzpLguhbdwNbuBdpe3CyrtNVWjtJYoCPbqiAs7W8Wa3IRGQaGNyo0/pkfzKu5pVja+3G3XWVVlXjoS3HUKaqwW2BTrhroBcAiDsfBLpaw81WAblMCo1WwP4r+eJrM4oqUVw79kcmlcDOwgz9vOxxePl4vD19YLNq0493AjgxoTupO0FBv0OHsce3Addb3LwdLMWJCmNCXAEAkX3cOb6NqBNhVyl1StUarbhJ9sEr+dBqBYNWrT/PZSO/TIUeTlbYPGcIiiuqEXMuSxzHFuRqA6lUAm9HSyTnl6NaI0Auk0Kt0aKyWiP+0nW0Mhd/qTW0SG9j3Gyvt7hxKZDuQ78kSHpRBVJr98E1hTFuod72kEqAsb1cxWMT+nlg+xMjEcKN5Yk6Fba4Uad0Jr1Y7MosKFfjYrbS4Plfazd9nzHEB7YW5vB1ssIDEX4AAHOZRBxzVnfs2QAfe7jWBq5zmbqxP3XHBrWEvaU55Ga6v14Mbt2HPtwfTylCtUaAhbkUXvbGH9840NcBx1+8E69O6V/vuJWc/38n6kwY3KhTOnilwODrA3W6OnNLq8Sv7x7oLR5fNC4YvdxtMW2wD8xkuh99H8fra1eF+zvC20H3S/Zchi4IOrUyuEkkEkwb7I3+3nYY5Nvx2xyRcei7yI+l6FqDA1xsTGZ8o5O13GRqIaLW43+1qFM6WLvnYpCrNZLyyrH/SgEWjgkCAPxxJgtaAQjzdUCPOouKutgosOPpMQbX8XW63hoS3sMR6UWViE8rxrkMfYtb0zNIG7P63gGtfi11Tvr9SlU1ugkxptBNSkRdC1vcqNOpqtbgRKpuJunSCbr9H48mF0BVo+s63V7bTTolzKvJa9VtcRvs5wif2ha35ALdUg6OrWxxo+6p7qQUAAgygRmlRNS1MLhRp3PyWhHUNVq42ykwsb8HXGwUqKrW4uS1YqQWVOBUajGkEmDyAM8mr9XX0w5SiW6JBBcbBbxrx7zpdzyou1UQUVM8b5jAYgozSomoa2FXKXU6+m7SEUEukEgkGBXsjJ/jM7E9PgMF5WrxOTfbpmeBBrvZ4IfHRogtJfoxbnqOt9BVSt2Pk7VcnJ0MsKuUiNoeW9yo09BoBaQVViDusm7j7uG1m8TrN8z+5lgadl3IAQDMHeHf7OsO6uEIr9rA5n3DDgdscaOWkEgkcLe/vhQMW9yIqK2xxY06hbiEXDz+1UlxCRAAGF67i8HoEFeYyySo1ggY5u+EZZN6IdyvdVv41G9xY3CjlvGws0BaYSXc7RSwUfCfWCJqW/xXhTqFn09loEKtgVwmha+TJaL6eYh7jXrYW+CbhbdBXSPgtkCnW1oF3tbCHHYWZlBW1QAw3CqIqDn03e6BLmxtI6K2x+BGncLZ2uU5PnowHHf0dqv3fGtb2Bri7WgFZZZuHbfWLsBL3Ze+u537fxJRe2BwI5NXpqrB1Xzd8hz9ve3b/f28HSxxsTa4sauUWurB2/ygqtZi3kh/Y5dCRF0QgxuZvPMZJRAE3VILrnX2AG0v+m2wJBLd1lVELeHjaIWX7u5n7DKIqIvirFIyefpu0tAOaG0Drk9QsLc0h4xbBBERkQlhcCOT1+HBrbbFjd2kRERkathVSibvbHptcPPpmOA2PNAZPd1t8M8BTW+ZRURE1JEY3MiklVZVixMTOqrFzdFajp1P394h70VERNQS7Colk3YuQze709vBEs427T8xgYiIyJQxuJFJO1c7vq2/t52RKyEiIjI+BjcyafqJCQN8HIxbCBERkQlgcCOTdlZsceuY8W1ERESmjMGNTFZJRTWSO3hiAhERkSljcCOTdSqtCADg72wFJ2uuqUZERMTgRibrZGoxAGBwD0fjFkJERGQiGNzIZJ1K1bW4DfJjcCMiIgIY3MhEabUC4sUWNwej1kJERGQqGNzIJCXmlqFUVQMruQy93G2NXQ4REZFJYHAjk3GtoBwXMnU7JZy4pusmHejjADMZf0yJiIgA7lVKJkKjFTB90yHkl6nw/WMjcLJ2fNtgPwfjFkZERGRCGNzIJCTllSG3VAUAeP6HM1DXaAFwRikREVFdDG5kEuLTisU/X84pE/88iMGNiIhIxMFDZBL0wa2nu414LMDFmgvvEhER1cHgRibhdG1wWxLZE+N7uwEAhvqztY2IiKgudpWS0VWqNbiUXQoACPN1wMggF3x9LBVTw7yNXBkREZFpYXAjozufWQKNVoCrrQKe9haQSCR49PYgY5dFRERkckyiq3TDhg3w9/eHhYUFIiIicPTo0UbP/fHHHzFkyBA4ODjA2toaYWFh+OKLLwzOEQQBK1euhKenJywtLREZGYnExMT2vg1qJf34tjBfB0gkEuMWQ0REZMKMHty2bduG6OhorFq1CidPnsTAgQMRFRWF3NzcBs93cnLCCy+8gEOHDuHMmTOYN28e5s2bhx07dojnrF27Fh988AE2bdqEI0eOwNraGlFRUaiqquqo26IWqBvciIiIqHESQRAEYxYQERGBoUOHYv369QAArVYLX19fLF68GM8//3yzrjF48GBMnjwZr776KgRBgJeXF5555hksXboUAFBSUgJ3d3ds3boVs2bNavJ6SqUS9vb2KCkpgZ2dXetvjgxcylbiX/89ikV3BOGhkQHi8VFv/o30okp8tSACI4NdjFghERGRcTQ3exi1xU2tVuPEiROIjIwUj0mlUkRGRuLQoUNNvl4QBMTGxiIhIQFjxowBACQnJyM7O9vgmvb29oiIiGj0miqVCkql0uBBbe+nUxnIL1PhP39fQbVGt8BufpkK6UWVkEiAUB97I1dIRERk2owa3PLz86HRaODu7m5w3N3dHdnZ2Y2+rqSkBDY2NpDL5Zg8eTL+85//4M477wQA8XUtuebq1athb28vPnx9fW/ltqgRJ1J021gVlKsRl5AH4PoyIEGuNrCzMDdWaURERJ2C0ce4tYatrS3i4+Nx7NgxvP7664iOjkZcXFyrr7d8+XKUlJSIj7S0tLYrlgAAqhoNzmSUiF//cCIdAPD5oWsAuGYbERFRcxh1ORAXFxfIZDLk5OQYHM/JyYGHh0ejr5NKpQgODgYAhIWF4eLFi1i9ejXGjh0rvi4nJweenp4G1wwLC2vwegqFAgqF4hbvhm7mXEYJ1DVaKMykUNVoEXspBz+fysCey3kwl0nwf2O4/AcREVFTjNriJpfLER4ejtjYWPGYVqtFbGwshg8f3uzraLVaqFS6DcoDAgLg4eFhcE2lUokjR4606JrUtk5c03WTjunpiv7edqjWCFj63WkAwNzh/vB3sTZmeURERJ2C0RfgjY6Oxty5czFkyBAMGzYM69atQ3l5OebNmwcAmDNnDry9vbF69WoAuvFoQ4YMQVBQEFQqFf744w988cUX2LhxIwBAIpFgyZIleO211xASEoKAgACsWLECXl5emDp1qrFus9s7Xju+bYifI+RmzjiXcQE1WgGOVuZYPD7EyNURERF1DkYPbjNnzkReXh5WrlyJ7OxshIWFISYmRpxckJqaCqn0esNgeXk5Hn/8caSnp8PS0hK9e/fGl19+iZkzZ4rnPPfccygvL8fChQtRXFyMUaNGISYmBhYWFh1+f6Sb/atvcRvi74gAFxu88cdFVGsEPH1nT9hbclICERFRcxh9HTdTxHXc2lZyfjnueDsOcjMpzr40AQozGb49noakvDIsndAL5rJOOUeGiIiozTQ3exi9xY26rmqNFjKJRGxtG+BtD4WZDAAwYwiXXCEiImopBjdqF+czS3DPhoNwtVXARqH7MQvnkh9ERES3hMGN2sXPpzKg1miRUVwpHgvvweBGRER0KxjcqF3sv1IAAJg73A/ZyioAuqVAiIiIqPUY3OiWVFVrcCylEKfTinFHbzf087JHXqkKF7N0+70uHh8CFxsubkxERNQWGNyo1TbvvYq3diZAXaPbMP7b4+nYvXQsDiblAwD6eNoxtBEREbUhrsNArfb54RSoa7TwsLOAtVyG1MIKxF7MwYEruuA2OsTFyBUSERF1LQxu1CrlqhqkFeomHvzx1Gj8a7gfAGDLgRTsT9QFt5HBDG5ERERticGNWiUxtwwA4GKjgJO1HHOG+0MmleDQ1QJkllRBLpNimL+TkaskIiLqWhjcqFUu55QCAHp52AAAvB0sEdXPXXw+3M8RlnKZUWojIiLqqhjcqFUuZ+uCW093W/HYvJEB4p9HcXwbERFRm2Nwo1ZJ0Le41QluQ/wcERHgBLlMatD6RkRERG2Dy4FQq+i7Snt6XA9uEokEW+YNRUllNTztLY1VGhERUZfF4EYtVlJRjRylCgAQ4mZj8JyV3AxWcv5YERERtQd2lVKLXc7VtbZ5O1jC1sLcyNUQERF1H60KbjU1Nfjrr7/w0UcfobRU90s8MzMTZWVlbVocmaYEcWKCTRNnEhERUVtqcZ/WtWvXMHHiRKSmpkKlUuHOO++Era0t3nzzTahUKmzatKk96iQT0tD4NiIiImp/LW5xe+qppzBkyBAUFRXB0vL6APR77rkHsbGxbVocmSZ9i1vdGaVERETU/lrc4rZv3z4cPHgQcrnc4Li/vz8yMjLarDAyTYIgXG9xY3AjIiLqUC1ucdNqtdBoNPWOp6enw9aWv8i7urwyFYoqqiGVAMFuHONGRETUkVoc3CZMmIB169aJX0skEpSVlWHVqlX4xz/+0Za1kQm6kqObgOLnbA0Lc25pRURE1JFa3FX6zjvvICoqCn379kVVVRXuv/9+JCYmwsXFBV9//XV71EgmJLWwAgDg52xl5EqIiIi6nxYHNx8fH5w+fRrffPMNzpw5g7KyMjz88MN44IEHDCYrUNeUVqQLbr6ODG5EREQdrVVL3JuZmeFf//pXW9dCnUBaYSUAwNeJIZ2IiKijtTi4ff755zd9fs6cOa0uhkwfW9yIiIiMp8XB7amnnjL4urq6GhUVFZDL5bCysmJw6+Kut7gxuBEREXW0Fs8qLSoqMniUlZUhISEBo0aN4uSELq6qWoP8Mt3m8j6O7ColIiLqaG2yyXxISAjWrFlTrzWOupb02m5SW4UZ7C25uTwREVFHa5PgBugmLGRmZrbV5cgE6btJfZysIJFIjFwNERFR99PiMW6//PKLwdeCICArKwvr16/HyJEj26wwMj3XJyawm5SIiMgYWhzcpk6davC1RCKBq6srxo0bh3feeaet6iITlFa7+C4nJhARERlHi4ObVqttjzqoExBnlLLFjYiIyCjabIwbdX1iVylb3IiIiIyiWS1u0dHRzb7gu+++2+piyLSlF9VOTuDiu0REREbRrOB26tSpZl2MMw27LmVVNUoqqwFwDTciIiJjaVZw2717d3vXQSZOPzHB2VoOa0WrtrglIiKiW8TfwHRTh5IK4ONoabCGGxERERlHq4Lb8ePH8e233yI1NRVqtdrguR9//LFNCiPjS8guxezNh2GjMMOYni4AOKOUiIjImFo8q/Sbb77BiBEjcPHiRfz000+orq7G+fPn8ffff8Pe3r49aiQjOZ1eDAAoU9Xgj7PZADijlIiIyJhaHNzeeOMNvPfee/j1118hl8vx/vvv49KlS5gxYwZ69OjRHjWSkSTllgEALM1l4jFOTCAiIjKeFge3pKQkTJ48GQAgl8tRXl4OiUSCp59+Gh9//HGbF0jGc6U2uC2b2AtzhvvB18kSY0JcjVwVERFR99XiMW6Ojo4oLS0FAHh7e+PcuXMIDQ1FcXExKioq2rxAMp7E2uDW08MWD40MMHI1RERE1OLgNmbMGOzatQuhoaGYPn06nnrqKfz999/YtWsXxo8f3x41khFUVWvEnRJC3GyNXA0REREBrQhu69evR1VVFQDghRdegLm5OQ4ePIhp06bhxRdfbPMCyTiS8sogCIC9pTlcbOTGLoeIiIjQiuDm5OQk/lkqleL5559v04LINOjHtwW72XBHDCIiIhPR4skJkZGR2Lp1K5RKZXvUQyZCP6M0xM3GyJUQERGRXouDW79+/bB8+XJ4eHhg+vTp2L59O6qrq9ujNjKixDotbkRERGQaWhzc3n//fWRkZODnn3+GtbU15syZA3d3dyxcuBB79uxpjxrJCK4wuBEREZmcFgc3QDe2bcKECdi6dStycnLw0Ucf4ejRoxg3blxb10dGUK3RIjm/HACDGxERkSm5pU3ms7Oz8c033+DLL7/EmTNnMGzYsLaqi4zoWkEFarQCLM1l8LLnTglERESmosUtbkqlElu2bMGdd94JX19fbNy4EXfffTcSExNx+PDh9qiROljdblKplDNKiYiITEWL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",
      "text/plain": [
       "<Figure size 700x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plot_evaluation([[ev for _, ev in report.get_evaluation(False)]], \"Overall test results\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "a9be40f2-7837-4569-a88a-b3b0603e336b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[{'accuracy': 0.10589}, {'accuracy': 0.12355999999999998}, {'accuracy': 0.14196}, {'accuracy': 0.13754}, {'accuracy': 0.16319}, {'accuracy': 0.16698}, {'accuracy': 0.18152999999999997}, {'accuracy': 0.20009000000000002}, {'accuracy': 0.20109}, {'accuracy': 0.21577000000000002}, {'accuracy': 0.21544000000000002}, {'accuracy': 0.22073}, {'accuracy': 0.23743000000000003}, {'accuracy': 0.22336}, {'accuracy': 0.22622}, {'accuracy': 0.23052}, {'accuracy': 0.25115000000000004}, {'accuracy': 0.26358}, {'accuracy': 0.2617299999999999}, {'accuracy': 0.27738999999999997}, {'accuracy': 0.28912}, {'accuracy': 0.28965}, {'accuracy': 0.28502}, {'accuracy': 0.29253}, {'accuracy': 0.29181}, {'accuracy': 0.30056}, {'accuracy': 0.29517000000000004}, {'accuracy': 0.29872}, {'accuracy': 0.30813999999999997}, {'accuracy': 0.30502}, {'accuracy': 0.31404}, {'accuracy': 0.32403}, {'accuracy': 0.32431}, {'accuracy': 0.33210999999999996}, {'accuracy': 0.32456}, {'accuracy': 0.32564}, {'accuracy': 0.33137999999999995}, {'accuracy': 0.33357000000000003}, {'accuracy': 0.32978}, {'accuracy': 0.34280999999999995}, {'accuracy': 0.32704000000000005}, {'accuracy': 0.33586}, {'accuracy': 0.3192900000000001}, {'accuracy': 0.34577}, {'accuracy': 0.33286000000000004}, {'accuracy': 0.34534999999999993}, {'accuracy': 0.34731999999999996}, {'accuracy': 0.33901}, {'accuracy': 0.34997}, {'accuracy': 0.32981}, {'accuracy': 0.3412800000000001}, {'accuracy': 0.34388}, {'accuracy': 0.35727}, {'accuracy': 0.36002}, {'accuracy': 0.36296000000000006}, {'accuracy': 0.36034}, {'accuracy': 0.3585900000000001}, {'accuracy': 0.36139}, {'accuracy': 0.35175}, {'accuracy': 0.35278}, {'accuracy': 0.36105}, {'accuracy': 0.35896}, {'accuracy': 0.35846999999999996}, {'accuracy': 0.35796}, {'accuracy': 0.36700000000000005}, {'accuracy': 0.36871}, {'accuracy': 0.36141999999999996}, {'accuracy': 0.3609}, {'accuracy': 0.35808000000000006}, {'accuracy': 0.35704}, {'accuracy': 0.36094000000000004}, {'accuracy': 0.35985000000000006}, {'accuracy': 0.35635000000000006}, {'accuracy': 0.35342}, {'accuracy': 0.35486000000000006}, {'accuracy': 0.35164999999999996}, {'accuracy': 0.3508}, {'accuracy': 0.35497999999999996}, {'accuracy': 0.35399}, {'accuracy': 0.35082}, {'accuracy': 0.35100000000000003}, {'accuracy': 0.35035999999999995}, {'accuracy': 0.35033000000000003}, {'accuracy': 0.35347}, {'accuracy': 0.35097000000000006}, {'accuracy': 0.3488}, {'accuracy': 0.3497}, {'accuracy': 0.34642}, {'accuracy': 0.34712}, {'accuracy': 0.34828}, {'accuracy': 0.34996}, {'accuracy': 0.34601000000000004}, {'accuracy': 0.35096999999999995}, {'accuracy': 0.34574}, {'accuracy': 0.3466599999999999}, {'accuracy': 0.34671}, {'accuracy': 0.34204}, {'accuracy': 0.34402}, {'accuracy': 0.34590000000000004}, {'accuracy': 0.34152}, {'accuracy': 0.32276999999999995}, {'accuracy': 0.33588999999999997}, {'accuracy': 0.33982999999999997}, {'accuracy': 0.3436100000000001}, {'accuracy': 0.34152000000000005}, {'accuracy': 0.35472000000000004}, {'accuracy': 0.3519}, {'accuracy': 0.34618000000000004}, {'accuracy': 0.34934}, {'accuracy': 0.3525600000000001}, {'accuracy': 0.34205}, {'accuracy': 0.35666}, {'accuracy': 0.35497}, {'accuracy': 0.35906}, {'accuracy': 0.35763999999999996}, {'accuracy': 0.33127000000000006}, {'accuracy': 0.34158000000000005}, {'accuracy': 0.33195}, {'accuracy': 0.34012000000000003}, {'accuracy': 0.35174}, {'accuracy': 0.35434}, {'accuracy': 0.3664}, {'accuracy': 0.3676}, {'accuracy': 0.34818}, {'accuracy': 0.35713}, {'accuracy': 0.3665}, {'accuracy': 0.37144}, {'accuracy': 0.38043000000000005}, {'accuracy': 0.37875}, {'accuracy': 0.37595999999999996}, {'accuracy': 0.37933}, {'accuracy': 0.37975}, {'accuracy': 0.37044}, {'accuracy': 0.37339}, {'accuracy': 0.37007}, {'accuracy': 0.37538}, {'accuracy': 0.38151999999999997}, {'accuracy': 0.37465999999999994}, {'accuracy': 0.37857999999999997}, {'accuracy': 0.37224999999999997}, {'accuracy': 0.37001000000000006}, {'accuracy': 0.37562}, {'accuracy': 0.37528}, {'accuracy': 0.38088}, {'accuracy': 0.37467}, {'accuracy': 0.37821000000000005}, {'accuracy': 0.38262999999999997}, {'accuracy': 0.37823}, {'accuracy': 0.38306}, {'accuracy': 0.38058}, {'accuracy': 0.38477}, {'accuracy': 0.38387}, {'accuracy': 0.38955}, {'accuracy': 0.38140999999999997}, {'accuracy': 0.38037}, {'accuracy': 0.38713}, {'accuracy': 0.38423000000000007}, {'accuracy': 0.38383}, {'accuracy': 0.38305}, {'accuracy': 0.38344}, {'accuracy': 0.38214000000000004}, {'accuracy': 0.3861}, {'accuracy': 0.38703000000000004}, {'accuracy': 0.38183}, {'accuracy': 0.38226}, {'accuracy': 0.38486}, {'accuracy': 0.38245}, {'accuracy': 0.38323}, {'accuracy': 0.38178000000000006}, {'accuracy': 0.37964}, {'accuracy': 0.38096}, {'accuracy': 0.38397}, {'accuracy': 0.383764}, {'accuracy': 0.38336}, {'accuracy': 0.383849}, {'accuracy': 0.381955}, {'accuracy': 0.383940990003}, {'accuracy': 0.382037}, {'accuracy': 0.384713}, {'accuracy': 0.38423000000000007}, {'accuracy': 0.383875}, {'accuracy': 0.37995999999999996}, {'accuracy': 0.379933}, {'accuracy': 0.379975}, {'accuracy': 0.38044}, {'accuracy': 0.38339}, {'accuracy': 0.38007}, {'accuracy': 0.38538}, {'accuracy': 0.38151999999999997}, {'accuracy': 0.38465999999999995}, {'accuracy': 0.3885799999999999}, {'accuracy': 0.38225}, {'accuracy': 0.38771000000000005}, {'accuracy': 0.38562}, {'accuracy': 0.38528}, {'accuracy': 0.38088}, {'accuracy': 0.38467}, {'accuracy': 0.386821}, {'accuracy': 0.38663}, {'accuracy': 0.386823}]\n"
     ]
    }
   ],
   "source": [
    "print([ev for _, ev in report.get_evaluation(False)])"
   ]
  },
  {
   "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
}
