{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "b512a328",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Using device: cuda:3\n"
     ]
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch_geometric.nn as pyg_nn\n",
    "import torch.nn.functional as F\n",
    "import torch_geometric.transforms as T\n",
    "from ogb.nodeproppred import PygNodePropPredDataset, Evaluator\n",
    "from prettytable import PrettyTable\n",
    "from torch import Tensor\n",
    "from torch_geometric.nn import GCNConv\n",
    "from torch_geometric.nn.conv import MessagePassing\n",
    "from torch_geometric.nn.inits import zeros\n",
    "from torch_geometric.typing import Adj, OptTensor, PairTensor\n",
    "from torch_sparse import SparseTensor, fill_diag, matmul, mul\n",
    "from typing import Optional, Tuple\n",
    "gpu_id = 3 # replace with index in [0, torch.cuda.device_count()]\n",
    "device = torch.device(\"cuda:\" + str(gpu_id) if torch.cuda.is_available() else \"cpu\")\n",
    "print('Using device:', device)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "288f9cbf",
   "metadata": {},
   "source": [
    "## Helpers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "da820b7c",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Logger object.\n",
    "class Logger(object):\n",
    "    def __init__(self, runs):\n",
    "        self.results = [[] for _ in range(runs)]\n",
    "\n",
    "    def add_result(self, run, result):\n",
    "        assert len(result) == 3\n",
    "        assert run >= 0 and run < len(self.results)\n",
    "        self.results[run].append(result)\n",
    "\n",
    "    def get_stats(self):\n",
    "        result = 100 * torch.tensor(self.results)\n",
    "\n",
    "        best_results = []\n",
    "        for r in result:\n",
    "            test = r[r[:, 1].argmax(), 2].item()\n",
    "            best_results.append(test)\n",
    "\n",
    "        return torch.tensor(best_results)\n",
    "\n",
    "# MLP with ReLU activations.\n",
    "class MLP(torch.nn.Module):\n",
    "    def __init__(self, num_layers, in_channels, hidden_channels, out_channels):\n",
    "        super(MLP, self).__init__()\n",
    "\n",
    "        self.lins = torch.nn.ModuleList()\n",
    "        self.lins.append(torch.nn.Linear(in_channels, hidden_channels))\n",
    "        self.bns = torch.nn.ModuleList()\n",
    "        self.bns.append(torch.nn.BatchNorm1d(hidden_channels))\n",
    "        for _ in range(num_layers - 2):\n",
    "            self.lins.append(torch.nn.Linear(hidden_channels, hidden_channels))\n",
    "            self.bns.append(torch.nn.BatchNorm1d(hidden_channels))\n",
    "        self.lins.append(torch.nn.Linear(hidden_channels, out_channels))\n",
    "\n",
    "        self.dropout = 0.5\n",
    "\n",
    "    def reset_parameters(self):\n",
    "        for lin in self.lins:\n",
    "            lin.reset_parameters()\n",
    "        for bn in self.bns:\n",
    "            bn.reset_parameters()\n",
    "\n",
    "    def forward(self, data):\n",
    "        x = data.x\n",
    "        for i, lin in enumerate(self.lins[:-1]):\n",
    "            x = lin(x)\n",
    "            x = self.bns[i](x)\n",
    "            x = F.relu(x)\n",
    "            x = F.dropout(x, p=self.dropout, training=self.training)\n",
    "        x = self.lins[-1](x)\n",
    "        return x\n",
    "    \n",
    "gcn_norm = pyg_nn.conv.gcn_conv.gcn_norm\n",
    "\n",
    "# GCNConv module modified to consider multiple convolutions.\n",
    "class GraphConv(MessagePassing):\n",
    "    _cached_edge_index: Optional[Tuple[Tensor, Tensor]]\n",
    "    _cached_adj_t: Optional[SparseTensor]\n",
    "\n",
    "    def __init__(\n",
    "        self, in_channels: int, out_channels: int, bias: bool = True,\n",
    "        num_convs: int = 1, cached: bool = False, **kwargs):\n",
    "        kwargs.setdefault('aggr', 'add')\n",
    "        super().__init__(**kwargs)\n",
    "\n",
    "        self.in_channels = in_channels\n",
    "        self.out_channels = out_channels\n",
    "        self.num_convs = num_convs\n",
    "        self.cached = cached\n",
    "        self._cached_edge_index = None\n",
    "        self._cached_adj_t = None\n",
    "\n",
    "        self.lin = nn.Linear(in_channels, out_channels, bias=False)\n",
    "\n",
    "        if bias:\n",
    "            self.bias = nn.Parameter(torch.Tensor(out_channels))\n",
    "        else:\n",
    "            self.register_parameter('bias', None)\n",
    "\n",
    "        self.reset_parameters()\n",
    "\n",
    "    def reset_parameters(self):\n",
    "        self.lin.reset_parameters()\n",
    "        zeros(self.bias)\n",
    "        self._cached_edge_index = None\n",
    "        self._cached_adj_t = None\n",
    "\n",
    "    def forward(self, x: Tensor, edge_index: Adj, edge_weight: OptTensor = None) -> Tensor:\n",
    "        if isinstance(edge_index, Tensor):\n",
    "            cache = self._cached_edge_index\n",
    "            if cache is None:\n",
    "                edge_index, edge_weight = gcn_norm(\n",
    "                    edge_index, edge_weight, x.size(self.node_dim),\n",
    "                    False, True)\n",
    "                if self.cached:\n",
    "                    self._cached_edge_index = (edge_index, edge_weight)\n",
    "            else:\n",
    "                edge_index, edge_weight = cache[0], cache[1]\n",
    "\n",
    "        elif isinstance(edge_index, SparseTensor):\n",
    "            cache = self._cached_adj_t\n",
    "            if cache is None:\n",
    "                edge_index = gcn_norm(\n",
    "                    edge_index, edge_weight, x.size(self.node_dim),\n",
    "                    False, True)\n",
    "                if self.cached:\n",
    "                    self._cached_adj_t = edge_index\n",
    "            else:\n",
    "                edge_index = cache\n",
    "\n",
    "        x = self.lin(x)\n",
    "\n",
    "        # propagate_type: (x: Tensor, edge_weight: OptTensor)\n",
    "        out = x\n",
    "        for i in range(self.num_convs):\n",
    "            out = self.propagate(edge_index, x=out, edge_weight=edge_weight, size=None)\n",
    "\n",
    "        if self.bias is not None:\n",
    "            out += self.bias\n",
    "\n",
    "        return out\n",
    "\n",
    "    def message(self, x_j: Tensor, edge_weight: OptTensor) -> Tensor:\n",
    "        return x_j if edge_weight is None else edge_weight.view(-1, 1) * x_j\n",
    "\n",
    "    def message_and_aggregate(self, adj_t: SparseTensor, x: Tensor) -> Tensor:\n",
    "        return matmul(adj_t, x, reduce=self.aggr)\n",
    "    \n",
    "# GCN with ReLU activations and sigmoidal output.\n",
    "class GCN(torch.nn.Module):\n",
    "    def __init__(self, num_layers, in_channels, hidden_channels, out_channels, num_convs):\n",
    "        super(GCN, self).__init__()\n",
    "\n",
    "        self.convs = torch.nn.ModuleList()\n",
    "        # self.convs.append(GCNConv(in_channels, hidden_channels, cached=True))\n",
    "        self.convs.append(GraphConv(in_channels, hidden_channels, num_convs=num_convs[0], cached=True))\n",
    "        self.bns = torch.nn.ModuleList()\n",
    "        self.bns.append(torch.nn.BatchNorm1d(hidden_channels))\n",
    "        for _ in range(num_layers - 2):\n",
    "            # self.convs.append(GCNConv(hidden_channels, hidden_channels, cached=True))\n",
    "            self.convs.append(\n",
    "                GraphConv(hidden_channels, hidden_channels, num_convs=num_convs[_+1], cached=True))\n",
    "            self.bns.append(torch.nn.BatchNorm1d(hidden_channels))\n",
    "        # self.convs.append(GCNConv(hidden_channels, out_channels, cached=True))\n",
    "        self.convs.append(GraphConv(hidden_channels, out_channels, num_convs=num_convs[-1], cached=True))\n",
    "\n",
    "        self.dropout = 0.5\n",
    "\n",
    "    def reset_parameters(self):\n",
    "        for conv in self.convs:\n",
    "            conv.reset_parameters()\n",
    "        for bn in self.bns:\n",
    "            bn.reset_parameters()\n",
    "\n",
    "    def forward(self, data):\n",
    "        x, adj_t = data.x, data.adj_t\n",
    "        for i, conv in enumerate(self.convs[:-1]):\n",
    "            x = conv(x, adj_t)\n",
    "            x = self.bns[i](x)\n",
    "            x = F.relu(x)\n",
    "            x = F.dropout(x, p=self.dropout, training=self.training)\n",
    "        return self.convs[-1](x, adj_t)\n",
    "    \n",
    "# Function to train a model, 1 epoch.\n",
    "def train(model, data, train_idx, optimizer):\n",
    "    model.train()\n",
    "\n",
    "    optimizer.zero_grad()\n",
    "    out = model(data)[train_idx]\n",
    "    loss = F.cross_entropy(out, data.y.squeeze(1)[train_idx])\n",
    "    loss.backward()\n",
    "    optimizer.step()\n",
    "\n",
    "    return loss.item()\n",
    "\n",
    "# Function to evaluate a model.\n",
    "@torch.no_grad()\n",
    "def test(model, data, split_idx, evaluator):\n",
    "    model.eval()\n",
    "\n",
    "    out =  torch.log_softmax(model(data), dim=-1)\n",
    "    y_pred = out.argmax(dim=-1, keepdim=True)\n",
    "\n",
    "    train_acc = evaluator.eval({\n",
    "        'y_true': data.y[split_idx['train']],\n",
    "        'y_pred': y_pred[split_idx['train']],\n",
    "    })['acc']\n",
    "    valid_acc = evaluator.eval({\n",
    "        'y_true': data.y[split_idx['valid']],\n",
    "        'y_pred': y_pred[split_idx['valid']],\n",
    "    })['acc']\n",
    "    test_acc = evaluator.eval({\n",
    "        'y_true': data.y[split_idx['test']],\n",
    "        'y_pred': y_pred[split_idx['test']],\n",
    "    })['acc']\n",
    "\n",
    "    return train_acc, valid_acc, test_acc"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c5482dd5-5833-40b8-a8fd-974a85855c5d",
   "metadata": {
    "tags": []
   },
   "source": [
    "## Learners and Plot Helpers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "supported-machinery",
   "metadata": {},
   "outputs": [],
   "source": [
    "model_types = [\n",
    "    'MLP2', # 2 layer MLP\n",
    "    '2L-01', # 2 layer GCN with 1 GC at layer 2\n",
    "    '2L-10', # 2 layer GCN with 1 GC at layer 1\n",
    "    '2L-02', # 2 layer GCN with 2 GCs at layer 2\n",
    "    '2L-20', # 2 layer GCN with 2 GCs at layer 1\n",
    "    '2L-11', # 2 layer GCN with 1 GC each at layers 1 and 2\n",
    "    '2L-03', # 2 layer GCN with 3 GCs at layer 2\n",
    "    '2L-30', # 2 layer GCN with 3 GCs at layer 1\n",
    "    '2L-12', # 2 layer GCN with 1 GC at layer 1 and 2 at layer 2\n",
    "    '2L-21', # 2 layer GCN with 2 GCs at layer 1 and 1 at layer 2\n",
    "]\n",
    "\n",
    "# Initialize all learning models.\n",
    "def init_learners(n_features, n_classes, hidden_channels=256):\n",
    "    learners = {}\n",
    "    learners['MLP2'] = MLP(2, n_features, hidden_channels, n_classes).to(device)\n",
    "    learners['2L-01'] = GCN(2, n_features, hidden_channels, n_classes, [0, 1]).to(device)\n",
    "    learners['2L-10'] = GCN(2, n_features, hidden_channels, n_classes, [1, 0]).to(device)\n",
    "    learners['2L-02'] = GCN(2, n_features, hidden_channels, n_classes, [0, 2]).to(device)\n",
    "    learners['2L-20'] = GCN(2, n_features, hidden_channels, n_classes, [2, 0]).to(device)\n",
    "    learners['2L-11'] = GCN(2, n_features, hidden_channels, n_classes, [1, 1]).to(device)\n",
    "    learners['2L-03'] = GCN(2, n_features, hidden_channels, n_classes, [0, 3]).to(device)\n",
    "    learners['2L-30'] = GCN(2, n_features, hidden_channels, n_classes, [3, 0]).to(device)\n",
    "    learners['2L-12'] = GCN(2, n_features, hidden_channels, n_classes, [1, 2]).to(device)\n",
    "    learners['2L-21'] = GCN(2, n_features, hidden_channels, n_classes, [2, 1]).to(device)\n",
    "    return learners\n",
    "\n",
    "alph = .8\n",
    "black = (.2, .2, .2, alph)\n",
    "red = (.8, .1, .1, alph)\n",
    "green = (.1, .8, .1, alph)\n",
    "blue = (0., .3, .8, alph)\n",
    "orange = (1., .6, .1, alph)\n",
    "colors=[black] + [red]*2 + [blue]*3 + [green]*4\n",
    "\n",
    "colors_std = []\n",
    "for color in colors:\n",
    "    color_std = (color[0], color[1], color[2], alph/3.)\n",
    "    colors_std.append(color_std)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "5a43c6e1",
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_for_models(ds_name, x_axis, y_axis, y_err, y_label, filename=None):\n",
    "    fig = plt.figure(figsize=(16,4), facecolor=[1,1,1])\n",
    "    y_axis = np.asarray(y_axis)\n",
    "    if y_err is None:\n",
    "        y_err = np.zeros(len(x_axis))\n",
    "    else:\n",
    "        y_err = np.asarray(y_err)\n",
    "    plt.ylim((0, 1.2*np.max(y_axis+y_err)))\n",
    "    plt.bar(x_axis, y_axis, color=colors, width=0.6)\n",
    "    plt.bar(x_axis, 2*y_err, color=colors_std, width=0.6, bottom=np.maximum(y_axis-y_err, np.zeros(y_axis.shape)))\n",
    "    plt.xticks(x_axis, [s.upper() for s in model_types])\n",
    "    txt_gap = .1*np.max(y_axis)\n",
    "    for i, v in enumerate(y_axis):\n",
    "        plt.text(i-.25, v + txt_gap, \" \"+str(round(v, 2)), color='black', va='center', fontweight='bold')\n",
    "    plt.xlabel('Model', fontsize=18)\n",
    "    plt.ylabel(y_label, fontsize=18)\n",
    "    plt.show()\n",
    "    if filename is not None:\n",
    "        fig.savefig(\"figures/custom-gcnconv/\" + filename, dpi=400, bbox_inches='tight')\n",
    "\n",
    "def plot_metrics(ds, x_axis, t_acc_avg, t_acc_std, t_acc_max):\n",
    "    fname = ds + '_test_acc_avg.pdf'\n",
    "    plot_for_models(ds, x_axis, t_acc_avg, t_acc_std, 'Accuracy (avg)', fname)\n",
    "    fname = ds + '_test_acc_max.pdf'\n",
    "    plot_for_models(ds, x_axis, t_acc_max, None, 'Accuracy (max)', fname)\n",
    "\n",
    "def print_metrics(ds, t_acc_avg, t_acc_std, t_acc_max):\n",
    "    t = PrettyTable(['Model Name', 'Accuracy', 'Accuracy (max)'])\n",
    "    for m, mtype in enumerate(model_types):\n",
    "        t.add_row([\n",
    "            mtype,\n",
    "            f'{t_acc_avg[m].item():.2f}' + u' \\u00B1 ' + f'{t_acc_std[m].item():.2f}',\n",
    "            f'{t_acc_max[m].item():.2f}'])\n",
    "    print('METRICS FOR DATASET:', ds.upper())\n",
    "    print(t)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d57c83d3-db7a-4870-a111-440c21d52338",
   "metadata": {},
   "source": [
    "## Train and test all models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "e2b2ebf1",
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Working on ogbn-arxiv: Classes = 40 | Points = 169343 | Features = 128\n",
      "Experiment completed.                                                                                 \n",
      "METRICS FOR DATASET: OGBN-ARXIV\n",
      "+------------+--------------+----------------+\n",
      "| Model Name |   Accuracy   | Accuracy (max) |\n",
      "+------------+--------------+----------------+\n",
      "|    MLP2    | 55.59 ± 0.20 |     55.91      |\n",
      "|   2L-01    | 68.11 ± 0.23 |     68.51      |\n",
      "|   2L-10    | 68.01 ± 0.20 |     68.37      |\n",
      "|   2L-02    | 71.46 ± 0.23 |     71.69      |\n",
      "|   2L-20    | 71.31 ± 0.34 |     71.74      |\n",
      "|   2L-11    | 71.43 ± 0.18 |     71.74      |\n",
      "|   2L-03    | 71.49 ± 0.21 |     71.81      |\n",
      "|   2L-30    | 71.61 ± 0.23 |     71.88      |\n",
      "|   2L-12    | 71.51 ± 0.30 |     71.84      |\n",
      "|   2L-21    | 71.30 ± 0.33 |     71.75      |\n",
      "+------------+--------------+----------------+\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAA7QAAAEOCAYAAAC5CVuUAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8QVMy6AAAACXBIWXMAAAsTAAALEwEAmpwYAAA6a0lEQVR4nO3deVxU9f7H8fcILrgvgRGoKKIiqKi4lbkh5lJ43bcUUy+ldZfUilJzKZM0M1NLcSUrSM3A63ZV8nrTMqNEr6JGqV0XFBU1N1Lw/P7o19wI0UFmmBl4PR8PHjnne873fL6fzpnDh7OZDMMwBAAAAACAkylh7wAAAAAAALgfFLQAAAAAAKdEQQsAAAAAcEoUtAAAAAAAp0RBCwAAAABwSq72DsAaHnjgAfn4+Ng7DAAAAACADRw/flznz5/PNb1IFLQ+Pj5KSkqydxgAAAAAABsIDg6+43QuOQYAAAAAOCUKWgAAAACAU6KgBQAAAAA4JQpaAAAAAIBToqAFAAAAADglCloAAAAAgFOioAUAAAAAOCUKWgAAAACAU6KgBQAAAAA4JQpaAAAAAIBToqAFAAAAADgluxa0c+bMUUBAgAIDAzVo0CBlZmYqIyNDoaGh8vPzU2hoqC5evGjPEAEAAAAADspuBe2pU6f07rvvKikpSQcOHFB2drbi4uIUFRWlkJAQpaamKiQkRFFRUfYKEQAAAADgwOx6hjYrK0s3btxQVlaWrl+/roceekgJCQkKDw+XJIWHhys+Pt6eIQJAgfj4+MhkMuX4GT58uCTp22+/VXBwsFxdXWUymZSUlHTP/l599VWZTCaVL18+x/SFCxfK19dXpUuXlq+vr7744gtbDAdFVGFsp2+++aYefPBBlSpVSjVq1NCUKVNsNBr7K4x8Hj58WB07dlS5cuXk6+urtWvX2mo4DsGaOc3r+3LTpk1q1KiRSpQoIZPJpPPnz9t6WHZjrXxOmTIlVz/JycmSilc+YV92K2i9vLw0fvx41axZU56enqpUqZK6dOmis2fPytPTU5Lk6emp9PT0Oy4fHR2t4OBgBQcH69y5c4UZOgBYbN68eYqNjVVsbKz69esnSWrWrJkk6caNGwoKClLTpk0t6uvgwYOaNWuWypQpk2P6unXrNHr0aHl7e2vBggUaNGiQbt68ad2BOBBr/SJ2twKruP0iVhjbqbu7uyZOnKiFCxeqYsWKmjp1qr788kvrDsRBFEY++/Xrp2+++UazZs1S1apVNXjwYKWlpVl3IA7EWjm92/fl9evX1a5dO/n6+tpuIA7CmtuoJHNfsbGx8vHxkVS88ikVzrEJeTDsJCMjw+jYsaORnp5u3Lx50+jZs6excuVKo1KlSjnmq1y58j37at68uY2iBADrCQwMNNzc3IyLFy/mmB4eHm5IMr755ps8l83OzjZat25t/PWvfzVq1apllCtXztz26KOPGmXLljUuXbpk3Lhxw1bhO4x169YZsbGxRmxsrNGvXz9DkjF37lzDMAzjiy++MEaOHGkEBwffM6dLly415s2bZyxdutRo2LChIcnYtWuXYRiGsWbNGmPMmDFG3bp1DUnGuXPnCmVsjsBW26lhGMbPP/9s/PTTT0aPHj0MScZXX31liyE4FFvk88KFC4Yko3PnzoZhGMbChQsNScbs2bNtNg5HUpCcWvJ92b59+2K13xckn5MnTzYkGdeuXTOysrLuOE9xyWdhHJuKu7xqPrudod22bZtq164td3d3lSxZUr1799aXX36p6tWrm//CmJaWJg8PD3uFCABWs2vXLh04cECDBg1S5cqV8738ggULlJaWpunTp+dqS0lJUalSpeTv76+yZcvq4Ycf1smTJ60QtWN64oknNHDgQA0cOFCHDh2Sm5ubhg0bJklq27atlixZooCAgHv2M2LECIWHh6tz586qXbu2JKlEiV8Pi3369NGCBQvk5eVlu4E4IFtup5I0dOhQ1apVSxs2bNC4cePUunXrAkbs2GyVzwoVKqhMmTI6fPiwUlNTtXPnTknSsWPHrBG2QytoTovb9+W9FDSfvylfvrzc3Nw0YMAAXb9+3XoBOpHCODbhzuyWnZo1a2r37t26fv26DMNQYmKi/P39FRYWppiYGElSTEyMevbsaa8QAcBqFi1aJEl65pln8r3spUuX9Morr+iFF17QmTNnlJWVJcMw9OOPP0qSfvnlF126dEnPPfecpk+frq+++kqRkZFWjd8RWeMXseJWYN2LLbdTSZo2bZrWrFmj5s2ba9GiRUpJSbFa7I7IVvksWbKkoqKilJaWpnr16ikhIUGScl2WXBQVJKdS8f2+zEtB89msWTMtWrRICQkJ6tGjh1atWqXZs2dbM0Snw7HJDgrxLHEur776qlG/fn0jICDAePLJJ43M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6Xyv/G9/+5u6du2qw4cPa9++ffL391dUVJRCQkKUmpqqkJAQRUVF5btfAAAAAEDRZ3FBK0kVKlTQyJEjtWPHDh09elRTpkzRrVu39NJLL6lWrVrq2rWrxX39/PPP+ve//62RI0dKkkqVKqXKlSsrISFB4eHhkqTw8HDFx8fnJ0QAAAAAQDGRr4L292rVqqVJkybp+++/10cffaRy5cpp69atFi9/9OhRubu766mnnlLTpk01atQoXbt2TWfPnpWnp6ckydPTU+np6XdcPjo6WsHBwQoODta5c+fudxgAAAAAACd13wXtlStXtGzZMnXo0EFDhw7Vzz//rICAAIuXz8rK0nfffafRo0dr7969KleuXL4uL46IiFBSUpKSkpLk7u5+P0MAAAAAADixfBW0hmFo8+bNGjx4sB588EGNGjVKhw4d0nPPPadvv/1W+/fvt7gvb29veXt7m18107dvX3333XeqXr260tLSJElpaWm8HggAAAAAcEcWP+V4/Pjx+vjjj3X27FmVLFnS/Aqf7t2739crfB588EHVqFFDR44cUf369ZWYmKiGDRuqYcOGiomJUWRkpGJiYtSzZ8989w0AAAAAKPosrkTffvtttWjRQhMnTtSgQYOs8iqdefPmaciQIbp586bq1Kmj5cuX6/bt2+rfv7+WLl2qmjVravXq1QVeDwAAAACg6LG4oE1JSVGDBg2suvKgoCAlJSXlmp6YmGjV9QAAAAAAih6L76G1djELAAAAAEBB5Pvm16SkJH399de6ePGibt++naPNZDJp0qRJVgsOAAAAAIC8WFzQ3rhxQ71799aWLVtkGIZMJpMMw5Ak878paAEAAAAAhcXiS46nTZumLVu2aMKECdq+fbsMw1BMTIw2bdqkRx99VC1atFBKSootYwUAAAAAwMzignbNmjXq16+fpk2bpsDAQEmSl5eXHnvsMW3btk03b97UihUrbBUnAAAAAAA5WHzJ8YkTJzR27FhJkouLiyTp5s2bv3bi6qpBgwbp/fff14wZM2wQpvP7+uuv7R2CVbRq1creIQAAAACApHwUtBUqVFBWVpb53yVKlNDp06fN7ZUqVdKZM2esH2ERMX/+fHuHYBUUtAAAAAAchcWXHPv6+ur777+X9OsZ2oCAAK1Zs0aSZBiG1q5dqxo1atgmSgAAAAAA/sDigrZz58769NNPlZ2d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zl/r1q1zLXv+/HmjU6dORt26dY1OnToZFy5cMLfVqlXLqFKlivn/w8GDBwtzWHZVkJyuXbvW8PLyMkqVKmV4eHgYXbp0Mbe9/vrrRp06dYx69eoZGzduLMwh2ZUtttHXXnvNKFu2rLm/Jk2aGGfPni3sodlNQXLq6+treHt7m5d7+umnzW1so9bd5/kete5+/8EHHxgNGzY0mjRpYjRt2tT47LPPCnlU9lOQfPbu3dsICAgwGjVqZDz++OPGyZMnzW3FdZ83DNvs98X52GSr41Jx/h41DMMwGcYdbg4AAAAAAMDBcQ8tAAAAAMApUdACAAAAAJwSBS0AAAAAwClR0AIAAAAAnBIFLQAAAADAKVHQAgBQhKxYsUImk+m+31v+r3/9SyaTSStWrLBqXAAA2AIFLQAAVvRbQWgymfTcc8/dcZ709HSVKlVKJpNJHTp0KNwAAQAoQihoAQCwgTJlyujjjz/WL7/8kqtt5cqVMgxDrq6udogMAICig4IWAAAb6NWrly5evKiEhIRcbcuXL1f37t1VunRpO0QGAEDRQUELAIANNGvWTE2aNNHy5ctzTN+zZ48OHjyop5566o7LxcfH65FHHlH58uVVvnx5PfLII3csiiVpyZIlatCggUqXLq26detq7ty5MgzjjvNevnxZL730kurWravSpUvL3d1dgwYN0tGjRws2UAAA7IhrnQAAsJGnnnpKY8eO1cmTJ+Xt7S1JWrZsmTw8PPT444/nmv+9997Ts88+qwYNGmjixInmhzP96U9/0qJFixQREWGe95133tHzzz+vJk2a6I033tD169c1a9YseXh45Or38uXLevjhh/Xf//5XI0aMUEBAgNLS0vTee++pVatWSkpKUq1atWyXCAAAbISCFgAAG3nyySf14osv6oMPPtArr7yiGzduKC4uTqNGjcp1/+zFixf14osvytfXV19//bUqVqwoSRo9erSaNm2qcePGqX///qpcubIuXbqkCRMmyN/fX19++aXKli0r6dcCukGDBrniePXVV3X06FHt3r1bTZo0MU8fPny4GjVqpMmTJ/NUYwCAU+KSYwAAbKRatWoKCwszF4tr167V5cuXNWLEiFzzbt26VdeuXdNf//pXczErSRUrVtRf/vIXXb16Vdu2bZMkbdmyRdevX9ezzz5rLmYlydvbW0OGDMnRr2EY+uijj9SuXTt5eXnp/Pnz5p9y5cqpdevW2rJliw1GDwCA7XGGFgAAG3rqqafUo0cP7dy5U8uWLVPLli3VsGHDXPMdO3ZMkhQQEJCrLTAwUJLM97v+9t87nY39Y9/nzp3ThQsXtGXLFrm7u98xxhIl+Ps2AMA5UdACAGBDjz32mLy8vDR16lRt375d77///h3ny+thTneb12Qy3bOf3z537txZL730ksXrAADAGVDQAgBgQy4uLho2bJhmzJghNzc3DRw48I7z+fr6SpIOHjyokJCQHG0pKSmSpDp16uSY99ChQ+rUqVOOeQ8dOpTjs7u7uypXrqyff/5ZnTt3LviAAABwIFxjBACAjT3zzDOaPHmyFi5cqEqVKt1xntDQUJUrV07z5s3TlStXzNOvXLmiefPmqXz58goNDTXP6+bmpgULFuj69evmeU+ePKmPP/44R78lSpTQkCFDtGfPHq1Zs+aO605PTy/oEAEAsAvO0AIAYGM1a9bUlClT7jpP5cqVNXPmTD377LNq1aqVhg8fLklasWKFfvjhBy1atMhcDFepUkWvvfaaxo8fr4cffljDhg3T9evXtXDhQvn5+Wnv3r05+p4+fbp27dql/v37q3///mrdurVKlSqln376SRs3blTz5s15yjEAwClR0AIA4CDGjBkjT09PzZo1S1OnTpUkNWnSRJ999pn+9Kc/5Zh33LhxKl++vN5++229/PLLqlGjhsaPH69KlSrleopypUqVtGvXLs2ePVurVq1SQkKCXF1d5e3trbZt22rUqFGFNUQAAKzKZOTnKRQAAAAAADgI7qEFAAAAADglCloAAAAAgFOioAUAAAAAOCUKWgAAAACAU6KgBQAAAAA4JQpaAAAAAIBToqAFAAAAADglCloAAAAAgFOioAUAAAAAOKX/A7BIVVQ8+L+nAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 1152x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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AAAAsCerC9qLFy+qcePGkiQXFxdJ0rlz5yztwcHB2rt3b4k2/sILL+iBBx7QkSNH9M033yggIECxsbEKDQ1Venq6QkNDFRsbW6J1AgAAAAAqBqsLWk9PT504cUKS5Obmprp16+rAgQOW9hMnTqhKFevHmPr999/1n//8R8OHD5ckVatWTXXr1lVycrKioqIkSVFRUUpKSrJ6nQAAAACAisPqCrRDhw7avXu35X14eLjmzZsnHx8fXblyRYsXL7YMYGSNn376Se7u7nr66af1zTffqH379lqwYIFOnTolT09PSVeL6IyMjBJ0BwAAAABQUVh9hnb48OFq2LChcnNzJUkzZ86Uq6urhg4dqmHDhsnFxUVz5syxesP5+fn6+uuvNWrUKO3bt09ubm4lurw4Li5OISEhCgkJUWZmptXLAQAAAADKB6vP0IaFhSksLMzyvlmzZvr++++VkpKiypUrq2vXrqpTp47VG/by8pKXl5flrG5kZKRiY2PVqFEjmc1meXp6ymw2y8PDo8jlo6OjFR0dLUkKCQmxersAAAAAgPLB6jO0RXFzc1NERIT69u1bomJWku644w41adJE3333nSQpJSVFrVq1UkREhBISEiRJCQkJ6tev3+2ECAAAAAAop6wfxckOFi1apMcff1yXLl1Ss2bNFB8frytXrmjgwIFatWqVvL29lZiYWJYhAgAAAAAcVIkK2vfee09LlixRenq6Tp8+XajdZDIpPz/f6vUFBwcrNTW10PSUlJSShAUAAAAAqICsLminT5+uyZMnq1GjRurSpYvq1atnz7gAAAAAALghqwvapUuXqkePHtq2bZuqVq1qz5gAAAAAALgpqweF+v333zVw4ECKWQAAAACAQ7C6oG3btq2OHz9uz1gAAAAAALCa1QXt9OnTtWzZMn399df2jAcAAAAAAKtYfQ9t9+7dtWrVKnXu3Fn33HOPmjZtqsqVK183j8lk0qpVq2weZHnw5JNPlnUINrF27dqyDgEAAAAAJJWgoP3yyy81dOhQ5efn69NPP9Wnn35aaB4KWgAAAABAabH6kuMXXnhBVatWVXJysrKysnTlypVCr4KCAnvGCgAAAACAhdVnaPfv368pU6bo4Ycftmc8AAAAAABYxeoztB4eHqpWrZo9YwEAAAAAwGpWF7TDhg3TunXrlJ+fb894AAAAAACwitWXHHft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\n",
      "text/plain": [
       "<Figure size 1152x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "dataset = PygNodePropPredDataset(name='ogbn-arxiv', root='./arxiv', transform=T.ToSparseTensor())\n",
    "data = dataset[0]\n",
    "data.adj_t = data.adj_t.to_symmetric()\n",
    "data = data.to(device)\n",
    "\n",
    "n_classes = dataset.num_classes\n",
    "n_points, n_features = data.num_nodes, data.num_features\n",
    "\n",
    "# Common variables.\n",
    "split_idx = dataset.get_idx_split()\n",
    "train_idx = split_idx['train'].to(device)\n",
    "\n",
    "n_trials = 10\n",
    "epochs = 500\n",
    "print('\\nWorking on ' + dataset.name + ': Classes =', n_classes, '| Points =', n_points, '| Features =', n_features)\n",
    "test_acc_avg = torch.zeros(len(model_types))\n",
    "test_acc_std = torch.zeros(len(model_types))\n",
    "test_acc_max = torch.zeros(len(model_types))\n",
    "learners = init_learners(n_features, n_classes)\n",
    "wspace = ' '*10\n",
    "for m, model_type in enumerate(model_types):\n",
    "    model = learners[model_type]\n",
    "    evaluator = Evaluator(name='ogbn-arxiv')\n",
    "    logger = Logger(n_trials)\n",
    "    for t in range(n_trials):\n",
    "        # Reset and train model.\n",
    "        model.reset_parameters()\n",
    "        optimizer = torch.optim.Adam(model.parameters(), lr=0.01)\n",
    "        for epoch in range(1, 1 + epochs):\n",
    "            loss = train(model, data, train_idx, optimizer)\n",
    "            result = test(model, data, split_idx, evaluator)\n",
    "            logger.add_result(t, result)\n",
    "            if epoch % 5 == 0:\n",
    "                train_acc, valid_acc, test_acc = result\n",
    "                print(f'\\rTrial: {t + 1} with {model_type}, '\n",
    "                      f'Epoch: {epoch:02d}, '\n",
    "                      f'Loss: {loss:.4f}, '\n",
    "                      f'Train: {100 * train_acc:.2f}%, '\n",
    "                      f'Valid: {100 * valid_acc:.2f}% '\n",
    "                      f'Test: {100 * test_acc:.2f}%', wspace, end='')\n",
    "    \n",
    "    # Get test metrics from accumulated test results.\n",
    "    test_acc_list = logger.get_stats()\n",
    "    test_acc_avg[m] = test_acc_list.mean().item()\n",
    "    test_acc_std[m] = test_acc_list.std().item()\n",
    "    test_acc_max[m] = test_acc_list.max().item()\n",
    "print('\\rExperiment completed.', wspace*8)\n",
    "x_axis = torch.arange(len(model_types))\n",
    "print_metrics(dataset.name, test_acc_avg, test_acc_std, test_acc_max)\n",
    "plot_metrics(dataset.name, x_axis, test_acc_avg, test_acc_std, test_acc_max)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1c9e3df8",
   "metadata": {},
   "source": [
    "## Plot and print metrics"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "2a235e19",
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "METRICS FOR DATASET: OGBN-ARXIV\n",
      "+------------+--------------+----------------+\n",
      "| Model Name |   Accuracy   | Accuracy (max) |\n",
      "+------------+--------------+----------------+\n",
      "|    MLP2    | 55.59 ± 0.20 |     55.91      |\n",
      "|   2L-01    | 68.11 ± 0.23 |     68.51      |\n",
      "|   2L-10    | 68.01 ± 0.20 |     68.37      |\n",
      "|   2L-02    | 71.46 ± 0.23 |     71.69      |\n",
      "|   2L-20    | 71.31 ± 0.34 |     71.74      |\n",
      "|   2L-11    | 71.43 ± 0.18 |     71.74      |\n",
      "|   2L-03    | 71.49 ± 0.21 |     71.81      |\n",
      "|   2L-30    | 71.61 ± 0.23 |     71.88      |\n",
      "|   2L-12    | 71.51 ± 0.30 |     71.84      |\n",
      "|   2L-21    | 71.30 ± 0.33 |     71.75      |\n",
      "+------------+--------------+----------------+\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 1152x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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z549tXHjRnXv3l0TJkzQLbfcUmqUY0nq1auXTQMEAAAAAKAsVhe0/fv3t/w9duxYmUymy9oNw5DJZGJgKAAAAABApbC6oE1KSrJnHAAAAAAAVIjVBW1MTIw94wAAAAAAoEKsHhQKAAAAAABHUm5Bm5aWds0r3b59+zUvCwAAAACANcotaO+66y7169dPGzdutGqgp/Pnz2v9+vXq3bu37r77bqs27ufnp5CQEIWGhiosLEySlJeXp4iICAUEBCgiIkL5+flWdgUAAAAAUJOUew/t3r17NXnyZEVFRenGG29URESEunbtKn9/fzVr1kyGYSgvL09ZWVn6/PPP9e9//1v5+fmKjIxURkaG1QHs2LFDN954o+V9fHy8wsPDFRcXp/j4eMXHx2vu3LnX1UkAAAAAQPVTbkHbvn17bd26VZ999pmWLVum1NRUrV69uszH9TRq1Ej333+/xo8fry5dulxXQKmpqdq5c6ekiwNR9enTh4IWAAAAAFDKVUc5vu2223TbbbeppKREe/bs0YEDB5SbmyuTySR3d3e1b99eHTt2VK1aFR9fymQyKTIyUiaTSY899phiY2N1/PhxeXl5SZK8vLyUk5NT5rIJCQlKSEiQJOXm5lZ42wAAAAAA52b1Y3tq166trl27qmvXrjbb+K5du3TzzTcrJydHERERatu2rdXLxsbGKjY2VpIs998CAAAAAGqOKn1sz8033yxJ8vDw0H333afdu3fL09NTZrNZkmQ2m+Xh4VGVIQIAAAAAHFSVFbRnz57V6dOnLX9v3bpV7du3V1RUlJKTkyVJycnJGjRoUFWFCAAAAABwYFZfcmxrx48f13333SdJKi4u1ogRI3TXXXepS5cuGjJkiBITE+Xj46OUlJSqChEAAAAA4MCqrKBt1aqVvv7661LTmzdvrrS0tCqICAAAAADgTKr0HloAAAAAAK4VBS0AAAAAwClZXdDOnj1bP//8sz1jAQAAAADAalYXtFOnTpWvr6/uvfdebdiwQSUlJfaMCwAAAACAK7K6oP388881ZswYffLJJxo8eLC8vb0VFxenb7/91p7xAQAAAABQJqsL2q5du2r58uUym81KSkpSYGCgXnnlFQUFBalXr15auXKlCgsL7RkrAAAAAAAWFR4UytXVVQ8//LA+/vhjffvtt3r66af1/fffa9SoUfLy8tKECROUkZFh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U6dOki4+4zYoKEg//fSTUlNTFRMTI0mKiYnRhg0brF4nAAAAAKDmsPqxPU8++aQ2b96s2bNn64033lDbtm1lMpl08OBB5eXl6Y477tCUKVOuKYijR49q79696tatm44fPy4vLy9JF4venJycMpdJSEhQQkKCJCk3N/eatgsAAAAAcF5Wn6GtW7eutm7dqvj4eHl7e2vv3r366quv1LJlS73yyivavn276tWrV+EAzpw5o8GDB2vBggVq1KiR1cvFxsYqPT1d6enpPBMXAAAAAGogq8/QSheL2qefflpPP/10me2///67XFxcrF7f+fPnNXjwYI0cOVL333+/JMnT01Nms1leXl4ym83y8PCoSIgAAAAAgBrC6jO0V7Jnzx5NmDBBN998s9XLGIahMWPGKCgoSJMnT7ZMj4qKUnJysiQpOTlZgwYNskWIAAAAAIBqpkJnaP8oLy9Pq1atUmJiovbv3y/DMBQYGGj18rt27dLKlSsVEhKi0NBQSdLLL7+suLg4DRkyRImJifLx8VFKSsq1hggAAAAAqMYqXND+61//0ttvv60PP/xQ586dU2BgoKZNm6bBgwcrODjY6vX07NlThmGU2ZaWllbRsAAAAAAANYxVBe2RI0eUlJSk5ORkHTt2TO7u7oqOjtb777+v2bNnW+5/BQAAAACgslzxHtr3339f4eHhCggI0CuvvKKwsDCtX79eP/30k6ZNm1buGVYAAAAAAOztimdoH3zwQbVq1UoLFizQiBEj1KxZM0ubyWSye3AAAAAAAJTnimdo69Wrp6NHjyo1NVWbN29WYWFhZcUFAAAAAMAVXbGg/eWXX7RgwQKdOHFCDz30kDw9PTVmzBj95z//4XJjAAAAAECVumJB26RJE02aNElfffWV0tPT9dBDD2nDhg3q27evevbsKZPJpFOnTlVWrAAAAAAAWFyxoP2jTp06aenSpfr555+1cuVKyyN6xo4dq9DQUM2aNUuZmZl2CxQAAAAAgD+yuqC9xMXFRSNGjFBaWpq+//57Pf/888rPz9eLL76oDh062CNGAAAAAABKqXBB+0d+fn6aOXOmjh49qk2bNvE8WgAAAABApbniY3usZTKZdNddd+muu+6yxeoAAAAAALiq6zpDCwAAAABAVaGgBQAAAAA4JQpaAAAAAIBToqAFAAAAADglCloAAAAAgFOioAUAAAAAOKUqK2hHjx4tDw8PtW/f3jItLy9PERERCggIUEREhPLz86sqPAAAAACAg6uygnbUqFHasmXLZdPi4+MVHh6urKwshYeHKz4+voqiAwAAAAA4uioraHv16qVmzZpdNi01NVUxMTGSpJiYGG3YsKEKIgMAAAAAOAOHuof2+PHj8vLykiR5eXkpJyeniiMCAAAAADiqOlUdwLVKSEhQQkKCJCk3N7eKowEAAAAAVDaHOkPr6ekps9ksSTKbzfLw8Ch33tjYWKWnpys9PV3u7u6VFSIAAAAAwEE4VEEbFRWl5ORkSVJycrIGDRpUxREBAAAAABxVlRW0w4cP12233abDhw/L29tbiYmJiouL07Zt2xQQEKBt27YpLi6uqsIDAAAAADi4KruHdvXq1WVOT0tLq+RIAAAAAADOyKEuOQYAAAAAwFoUtAAAAAAAp0RBCwAAAABwShS0AAAAAACnREELAAAAAHBKFLQAAAAAAKdEQQsAAAAAcEoUtAAAAAAAp0RBCwAAAABwShS0AAAAAACnREELAAAAAHBKFLQAAAAAAKdEQQsAAAAAcEoUtAAAAAAAp0RBCwAAAABwShS0AAAAAACnREELAAAAAHBKDlnQbtmyRW3atFHr1q0VHx9f1eEAAAAAAByQwxW0JSUlmjhxojZv3qwDBw5o9erVOnDgQFWHBQAAAABwMA5X0O7evVutW7dWq1atVK9ePQ0bNkypqalVHRYAAAAAwMHUqeoA/uynn35Sy5YtLe+9vb31xRdflJovISFBCQkJkqRDhw4pLCys0mJ0VLm5uXJ3d7frNmpSnisjn5IkcmpbNSifUmUd93ZdvUMhn7ZHTm2rsr6bwlRzklop+yj5tDlyalvOkM+jR4+WOd3hClrDMEpNM5lMpabFxsYqNja2MkJyGmFhYUpPT6/qMKoN8ml75NT2yKltkU/bI6e2RT5tj5zaFvm0PXJ6ZQ53ybG3t7eys7Mt748dO6abb765CiMCAAAAADgihytou3TpoqysLB05ckTnzp3TmjVrFBUVVdVhAQAAAAAcjMNdclynTh0tWbJEd955p0pKSjR69GgFBwdXdVhOgUuwbYt82h45tT1yalvk0/bIqW2RT9sjp7ZFPm2PnF6ZySjrplUAAAAAABycw11yDAAAAACANShoAQAAAABOiYLWgZlMJj300EOW98XFxXJ3d9c999wjSXrnnXc0adKkUsv5+fkpJCREHTp0UGRkpH755RcVFBRo4MCBatu2rYKDgxUXF1dp/XAE2dnZ6tu3r4KCghQcHKyFCxdKkkaNGqW1a9decdm8vDxFREQoICBAERERys/PlySdOHFCffv2VcOGDcv8/1DdXU9OU1JSFBwcrFq1apUahn7OnDlq3bq12rRpo3/96192i9/R2GMf3bZtmzp37qyQkBB17txZ//73v+3eD0dyPTn9+9//rrZt2+rWW2/Vfffdp5MnT1ra2Edtd8zzOWr743737t0KDQ1VaGioOnTooPXr19u9H47ievI5depU3XrrrQoNDVVkZKR+/vlnS1tNPeYl+xz3Nfm7yR7fSzX9c1SioHVobm5u2r9/vwoLCyVd/ABo0aKFVcvu2LFDX3/9tcLCwvTyyy9Lkp566ikdOnRIe/fu1a5du7R582a7xe5o6tSpo3nz5ungwYP6/PPPtXTpUh04cMCqZePj4xUeHq6srCyFh4crPj5eklS/fn299NJLeu211+wZusO6npy2b99e69atU69evS6bfuDAAa1Zs0aZmZnasmWLJkyYoJKSEnuE73DssY/eeOON+r//+z998803Sk5OvuwfyGqC68lpRESE9u/fr3379ikwMFBz5syRxD5q62Oez1HbH/ft27dXenq6MjIytGXLFj322GMqLi62ZzccxvXk8+9//7v27dunjIwM3XPPPZo5c6akmn3MS/Y57mvyd5M9vpdq+ueoREHr8AYMGKCPPvpIkrR69WoNHz68Qsv36tVL3333nRo0aKC+fftKkurVq6dOnTrp2LFjNo/XUXl5ealTp06SpBtuuEFBQUH66aefrFo2NTVVMTExkqSYmBht2LBB0sV/cOjZs6fq169vl5gd3fXkNCgoSG3atCk1PTU1VcOGDZOLi4tuueUWtW7dWrt377Zp3I7KHvtox44dLc/xDg4OVlFRkX7//XfbB++grienkZGRqlPn4oMAunfvbvm8ZB+17THP56jtj/sGDRpY9t2ioiKZTCbbB+6griefjRo1svx99uxZS95q8jEv2ee4r8nfTfb4Xqrpn6MSBa3DGzZsmNasWaOioiLt27dP3bp1q9DyGzduVEhIyGXTTp48qf/7v/9TeHi4LUN1GkePHtXevXutzuXx48fl5eUl6eIHUU5Ojj3Dc0oVzWl5fvrpJ7Vs2dLy3tvb2+oP+urEHvvoBx98oI4dO8rFxcWmsTqL69lH3377bQ0YMEAS++gltjrm8T+2PO6/+OILBQcHKyQkRMuXL7f8CK5JrmUfff7559WyZUu99957ljO0HPP/Y4/jviZ/N9nqewkUtA7v1ltv1dGjR7V69WrdfffdVi/Xt29fhYaG6rffftOzzz5rmV5cXKzhw4fr8ccfV6tWrewRskM7c+aMBg8erAULFlz2r7G4drbMaVlPEatJZxck++yjmZmZeuaZZ7RixQqbrM/ZXE9OZ8+erTp16mjkyJGS2EclPkftwdY57datmzIzM/Xll19qzpw5KioqskGUzuNa8zl79mxlZ2dr5MiRWrJkiSSO+Uv4brItW34vgYLWKURFRempp56q0OXGO3bsUEZGht599101adLEMj02NlYBAQH629/+ZvtAHdz58+c1ePBgjRw5Uvfff3+58z3yyCMKDQ21/AOCp6enzGazJMlsNsvDw6NS4nUG15rT8nh7eys7O9vy/tixY5bLkmoCe+yjx44d03333ad3331X/v7+9u2AA7qefTQ5OVkbN27Ue++9Z/kByz5q22Me9v1uCgoKsozHUVPYYh8dMWKEPvjgA0kc85J9jvua/N1k6+8lSDXvGhQnNHr0aDVu3FghISHauXPnNa/nhRde0KlTp/TWW2/ZLjgnYRiGxowZo6CgIE2ePPmK8yYlJV32PioqSsnJyYqLi1NycrIGDRpkz1CdxvXktDxRUVEaMWKEJk+erJ9//llZWVnq2rWrLcJ1ePbYR0+ePKmBAwdqzpw5uv322+0Wu6O6npxu2bJFc+fO1ccff6wGDRpYprOP2vaYr+nscdwfOXJELVu2VJ06dfTjjz/q8OHD8vPzs1cXHMr15DMrK0sBAQGSpA8//FBt27aVVLOPeck+x31N/m6yx/cSJBlwWG5ubqWm7dixwxg4cKBhGIaRlJRkuLm5GS1atLC8srOzDV9fXyM3N/ey5bKzsw1JRtu2bY0OHToYHTp0MN58881K6Ycj+OSTTwxJRkhIiKX/H330kRETE2M0a9bMkr/u3buXWvbXX381+vXrZ7Ru3dro16+fceLECUubr6+v0bRpU8v/h8zMzMrsVpW6npyuW7fOaNGihVGvXj3Dw8PDiIyMtLTNmjXLaNWqlREYGGhs2rSpMrtUpeyxj7700ktGgwYNLOvr0KGDcfz48cruWpW5npz6+/sb3t7eluUee+wxSxv7qG2PeT5HbXvcv/vuu0a7du2MDh06GB07djTWr19fyb2qOteTz/vvv98IDg42QkJCjHvuucc4duyYpa2mHvOGYZ/jviZ/N9nre6kmf44ahmGYDKOMmwMAAAAAAHBw3EMLAAAAAHBKFLQAAAAAAKdEQQsAAAAAcEoUtAAAAAAAp0RBCwAAAABwShS0AABUI++8845MJtM1P7d8586dMplMeuedd2waFwAA9kBBCwCADV0qCE0mkyZNmlTmPDk5OapXr55MJpP69OlTuQECAFCNUNACAGAH9evX1/vvv6/ff/+9VNvKlStlGIbq1KlTBZEBAFB9UNACAGAH9913n/Lz85WamlqqLSkpSXfffbdcXFyqIDIAAKoPCloAAOygU6dO6tChg5KSki6bvnv3bmVmZuqRRx4pc7kNGzbo9ttvV8OGDdWwYUPdfvvtZRbFkvTWW2+pbdu2cnFxUevWrbVw4UIZhlHmvKdOndIzzzyj1q1by8XFRe7u7ho+fLh++OGH6+soAABViGudAACwk0ceeUSTJ0/WsWPH5O3tLUl6++235eHhoXvuuafU/MuWLdPEiRPVtm1bvfDCC5bBmf7yl79oxYoVio2Ntcy7YMECPfnkk+rQoYNefvllFRQU6NVXX5WHh0ep9Z46dUo9evTQ//t//0+jR49WcHCwzGazli1bpm7duik9PV2+vr72SwQAAHZCQQsAgJ08+OCDevrpp/Xuu+/queeeU2FhodasWaOxY8eWun82Pz9fTz/9tPz9/fXFF1+oUaNGkqTx48erY8eOmjJlioYMGaImTZro5MmTev755xUUFKT//ve/atCggaSLBXTbtm1LxfHiiy/qhx9+0Oeff64OHTpYpo8aNUohISGaNm0aoxoDAJwSlxwDAGAnzZs3V1RUlKVYXLdunU6dOqXRo0eXmnfbtm06e/asHn/8cUsxK0mNGjXSX//6V505c0bbt2+XJG3dulUFBQWaOHGipZiVJG9vb40cOfKy9RqGoffee0+9evVSixYt9Ouvv1pebm5u6t69u7Zu3WqH3gMAYH+coQUAwI4eeeQRDRw4UJ9++qnefvttde3aVe3atSs135EjRyRJwcHBpdrat28vSZb7XS/9t6yzsX9ed25urk6cOKGtW7fK3d29zBhr1eLftwEAzomCFgAAO7rzzjvVokULzZgxQzt27NAbb7xR5nzlDeZ0pXlNJtNV13Ppff/+/fXMM89YvQ0AAJwBBS0AAHZUu3ZtPfzww5ozZ45cXV01bNiwMufz9/eXJGVmZio8PPyytgMHDkiSWrVqddm8Bw8eVL9+/S6b9+DBg5e9d3d3V5MmTfTbb7+pf//+198hAAAcCNcYAQBgZ+PGjdO0adO0fPlyNW7cuMx5IiIi5ObmpsWLF+v06dOW6adPn9bixYvVsGFDRUREWOZ1dXXV0qVLVVBQYJn32LFjev/99y9bb61atTRy5Ejt3r1ba9euLXPbOTk519tFAACqBGdoAQCwMx8fH02fPv2K8zRp0kSvvPKKJk6cqG7dumnUqFGSpHfeeUffffedVqxYYSmGmzZtqpdeeklPPfWUevTooYcfflgFBQVavny5AgICtHfv3svWPXv2bO3atUtDhgzRkCFD1L17d9WrV08//vijNm3apM6dOzPKMQDAKVHQAgDgICZMmCAvLy+9+uqrmjFjhiSpQ4cOWr9+vf7yl79cNu+UKVPUsGFDvf7663r22WfVsmVLPfXUU2rcuHGpUZQbN26sXbt2ad68efrnP/+p1NRU1alTR97e3urZs6fGjh1bWV0EAMCmTEZFRqEAAAAAAMBBcA8tAAAAAMApUdACAAAAAJwSBS0AAAAAwClR0AIAAAAAnBIFLQAAAADAKVHQAgAAAACcEgUtAAAAAMApUdACAAAAAJwSBS0AAAAAwCn9fxPC+tnH7pSnAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 1152x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "x_axis = torch.arange(len(model_types))\n",
    "print_metrics(dataset.name, test_acc_avg, test_acc_std, test_acc_max)\n",
    "plot_metrics(dataset.name, x_axis, test_acc_avg, test_acc_std, test_acc_max)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6112ea61",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
