{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Requirement already satisfied: ogb in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (1.1.1)\n",
      "Requirement already satisfied: tqdm>=4.29.0 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from ogb) (4.46.0)\n",
      "Requirement already satisfied: torch>=1.2.0 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from ogb) (1.4.0)\n",
      "Requirement already satisfied: scikit-learn>=0.20.0 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from ogb) (0.23.1)\n",
      "Requirement already satisfied: urllib3>=1.24.0 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from ogb) (1.25.9)\n",
      "Requirement already satisfied: numpy>=1.16.0 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from ogb) (1.18.1)\n",
      "Requirement already satisfied: six>=1.12.0 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from ogb) (1.14.0)\n",
      "Requirement already satisfied: pandas>=0.24.0 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from ogb) (1.0.3)\n",
      "Requirement already satisfied: joblib>=0.11 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from scikit-learn>=0.20.0->ogb) (0.15.1)\n",
      "Requirement already satisfied: scipy>=0.19.1 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from scikit-learn>=0.20.0->ogb) (1.4.1)\n",
      "Requirement already satisfied: threadpoolctl>=2.0.0 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from scikit-learn>=0.20.0->ogb) (2.0.0)\n",
      "Requirement already satisfied: pytz>=2017.2 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from pandas>=0.24.0->ogb) (2020.1)\n",
      "Requirement already satisfied: python-dateutil>=2.6.1 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from pandas>=0.24.0->ogb) (2.8.1)\n",
      "Requirement already satisfied: POT in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (0.7.0)\n",
      "Requirement already satisfied: cython>=0.23 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from POT) (0.29.19)\n",
      "Requirement already satisfied: numpy>=1.16 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from POT) (1.18.1)\n",
      "Requirement already satisfied: scipy>=1.0 in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (from POT) (1.4.1)\n",
      "Requirement already satisfied: prettytable in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (0.7.2)\n",
      "Requirement already satisfied: tqdm in /home/user/miniconda/envs/py36/lib/python3.6/site-packages (4.46.0)\n"
     ]
    }
   ],
   "source": [
    "# install required packages\n",
    "!pip install ogb\n",
    "!pip install POT\n",
    "!pip install prettytable\n",
    "!pip install tqdm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "ogb version 1.1.1\n"
     ]
    }
   ],
   "source": [
    "# import required modules\n",
    "import ogb; print('ogb version {}'.format(ogb.__version__)) # make sure the version is =>1.1.1.\n",
    "from ogb.graphproppred import PygGraphPropPredDataset\n",
    "from ogb.graphproppred import Evaluator\n",
    "from ogb.graphproppred.mol_encoder import AtomEncoder, BondEncoder\n",
    "\n",
    "import torch\n",
    "import torch.nn.functional as F\n",
    "\n",
    "import torch_geometric\n",
    "from torch_geometric.data import DataLoader\n",
    "import torch_geometric.transforms as T\n",
    "\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from sklearn.decomposition import PCA\n",
    "from sklearn.cluster import KMeans\n",
    "from sklearn.ensemble import RandomForestClassifier\n",
    "from sklearn.model_selection import PredefinedSplit\n",
    "from sklearn.model_selection import GridSearchCV\n",
    "\n",
    "import ot\n",
    "\n",
    "from diffusion_layer_multiple_nonneg_edge_features import Diffusion_layer\n",
    "\n",
    "import numpy as np\n",
    "import random\n",
    "import itertools\n",
    "from collections import defaultdict\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "from tqdm.notebook import tqdm\n",
    "from prettytable import PrettyTable"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "# set the random seed\n",
    "random_seed = 55\n",
    "random.seed(random_seed)\n",
    "np.random.seed(random_seed)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "# of graphs = 41127\n",
      "# of classes = 2\n",
      "# of node features = 9\n",
      "# of edge features = 3\n",
      "# of tasks = 1\n"
     ]
    }
   ],
   "source": [
    "# load the dataset\n",
    "dataset = PygGraphPropPredDataset(name=\"ogbg-molhiv\")\n",
    "\n",
    "print('# of graphs = {0}\\n# of classes = {1}\\n# of node features = {2}\\n# of edge features = {3}'.\\\n",
    "         format(len(dataset), dataset.num_classes, dataset.num_node_features, dataset.num_edge_features))\n",
    "\n",
    "if isinstance(dataset, PygGraphPropPredDataset): # OGB datasets\n",
    "    print('# of tasks = {}'.format(dataset.num_tasks))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define the Diffusion class\n",
    "class Diffusion(torch.nn.Module):\n",
    "    def __init__(self, num_hidden_layers, final_node_embedding='final'):\n",
    "        super(Diffusion, self).__init__()\n",
    "            \n",
    "        # create the hidden layers\n",
    "        self.layers = torch.nn.ModuleList([Diffusion_layer() for _ in range(num_hidden_layers)])\n",
    "        \n",
    "        self.final_node_embedding = final_node_embedding\n",
    "            \n",
    "    def forward(self, data):\n",
    "        x, edge_index, edge_attr = data.x, data.edge_index, data.edge_attr\n",
    "        \n",
    "        out_concat = [x]\n",
    "        for i in range(len(self.layers)):\n",
    "            x = self.layers[i](x, edge_index, edge_attr)\n",
    "            out_concat.append(x)\n",
    "\n",
    "        if self.final_node_embedding == 'final':\n",
    "            return x\n",
    "        elif self.final_node_embedding == 'avg':\n",
    "            return torch.mean(torch.stack(out_concat), dim=0)\n",
    "        elif self.final_node_embedding == 'concat':\n",
    "            return torch.cat(out_concat, dim=1)\n",
    "        else:\n",
    "            raise NotImplementedError()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define the bond encoder function\n",
    "def BondEncoderOneHot(edge_attr):\n",
    "    \"\"\"\n",
    "    Encode each of the three bond features using one-hot encoding, and concatenate them together\n",
    "    Each bond has three features:\n",
    "        Feature 0 (possible_bond_type_list) has 5 categories.\n",
    "        Feature 1 (possible_bond_stereo_list) has 6 categories.\n",
    "        Feature 2 (possible_is_conjugated_list) has 2 categories.\n",
    "    \n",
    "    Source: https://github.com/snap-stanford/ogb/blob/master/ogb/utils/features.py\n",
    "    \n",
    "    Input: |E| x 3 edge feature matrix\n",
    "    Output: |E| x (5 + 6 + 2) = |E| x 13 one-hot edge feature matrix\n",
    "    \"\"\"\n",
    "    \n",
    "    num_feature_categories = [5, 6, 2]\n",
    "    all_one_hot_features = []\n",
    "    for col in range(3):\n",
    "        one_hot_features = torch.nn.functional.one_hot(edge_attr[:, col], num_feature_categories[col])\n",
    "        all_one_hot_features.append(one_hot_features)\n",
    "    all_one_hot_features = torch.cat(all_one_hot_features, dim=1)\n",
    "    \n",
    "    return all_one_hot_features"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "# define the WEGL pipeline\n",
    "def WEGL(dataset, # dataset object\n",
    "         num_hidden_layers, # number of diffusion layers\n",
    "         node_embedding_sizes, # node embedding dimensionality created by the AtomEncoder module\n",
    "         final_node_embedding, # final node embedding type $\\in$ {'concat', 'avg', 'final'}\n",
    "         num_pca_components=20, # number of PCA components applied on node embeddings. -1 means no PCA.\n",
    "         num_experiments=10, # number of experiments with different random seeds\n",
    "         classifiers=['RF'], # list of downstream classifiers (currently random forest ('RF') only; other classifiers, e.g., SVM, can be added if needed)\n",
    "         device='cpu', # the device to run the diffusion over\n",
    "        ):\n",
    "    \n",
    "    # Create data loaders\n",
    "    split_idx = dataset.get_idx_split() # train/val/test split\n",
    "    loader_dict = {}\n",
    "    for phase in split_idx:\n",
    "        batch_size = 32\n",
    "        loader_dict[phase] = DataLoader(dataset[split_idx[phase]], batch_size=batch_size, shuffle=False)\n",
    "    \n",
    "    # prepare the output table\n",
    "    results_table = PrettyTable()\n",
    "    results_table.title = 'Final ROC-AUC(%) results for the {0} dataset with \\'{1}\\' node embedding and one-hot 13-dim edge embedding'.\\\n",
    "                               format(dataset.name, final_node_embedding)\n",
    "\n",
    "    results_table.field_names = ['Classifier', '# Diffusion Layers', 'Node Embedding Size',\n",
    "                                 'Train.', 'Val.', 'Test']\n",
    "\n",
    "    n_jobs = 14\n",
    "    verbose = 0\n",
    "    \n",
    "    for L, F in itertools.product(num_hidden_layers, node_embedding_sizes):\n",
    "        print('*' * 100)\n",
    "        print('# diffusion layers = {0}, node embedding size = {1}, node embedding mode: {2}'.\\\n",
    "              format(L, F, final_node_embedding))\n",
    "        \n",
    "        # create the diffusion object\n",
    "        diffusion = Diffusion(num_hidden_layers=L,\n",
    "                              final_node_embedding=final_node_embedding).to(device)\n",
    "        diffusion.eval()\n",
    "        \n",
    "        # create the node encoder\n",
    "        node_feature_encoder = AtomEncoder(F).to(device)\n",
    "        node_feature_encoder.eval()\n",
    "\n",
    "        phases = list(loader_dict.keys()) # determine different partitions of data ('train', 'valid' and 'test')\n",
    "\n",
    "        # pass the all the graphs in the data through the GNN\n",
    "        X = defaultdict(list)\n",
    "        Y = defaultdict(list)\n",
    "\n",
    "        for phase in phases:\n",
    "            print('Now diffusing the ' + phase + ' data ...')\n",
    "            for i, batch in enumerate(tqdm(loader_dict[phase])):\n",
    "                batch = batch.to(device)\n",
    "\n",
    "                # encode node features\n",
    "                batch.x = node_feature_encoder(batch.x)\n",
    "                \n",
    "                # encode edge features\n",
    "                batch.edge_attr = BondEncoderOneHot(batch.edge_attr)\n",
    "                \n",
    "                # add virtual nodes\n",
    "                batch_size = len(batch.y)\n",
    "                num_original_nodes = batch.x.size(0)\n",
    "                batch.batch = torch.cat((batch.batch, torch.Tensor(range(batch_size)).to(batch.batch.dtype)), dim=0)\n",
    "                \n",
    "                # make the initial features of all virtual nodes zero\n",
    "                batch.x = torch.cat((batch.x, batch.x.new_zeros(batch_size, batch.x.size(1))), dim=0)\n",
    "                \n",
    "                \n",
    "                \n",
    "                # add edges between all nodes in each graph and the virtual node for that graph\n",
    "                for g in range(batch_size):\n",
    "                    node_indices = np.where(batch.batch == g)[0][:-1] # last node is the virtual node\n",
    "                    virtual_edges_one_way = np.array([node_indices, (num_original_nodes + g) * np.ones_like(node_indices)])\n",
    "                    virtual_edges_two_ways = np.concatenate((virtual_edges_one_way,\n",
    "                                                             np.take(virtual_edges_one_way, [1,0], axis=0)),\n",
    "                                                            axis=1)\n",
    "                    \n",
    "                    batch.edge_index = torch.cat((batch.edge_index,\n",
    "                                                  torch.Tensor(virtual_edges_two_ways).to(batch.edge_index.dtype)),\n",
    "                                                 dim=1)\n",
    "                    \n",
    "                    # make the initial edge features of all edges to/from virtual nodes all 1 / number of graph nodes\n",
    "                    batch.edge_attr = torch.cat((batch.edge_attr, \n",
    "                                                 batch.edge_attr.new_ones(2 * len(node_indices),\n",
    "                                                                          batch.edge_attr.size(1)) / len(node_indices)),\n",
    "                                                dim=0)\n",
    "                \n",
    "                z = diffusion(batch)\n",
    "\n",
    "                batch_indices = batch.batch.cpu()\n",
    "                for b in range(batch_size):\n",
    "                    node_indices = np.where(batch_indices == b)[0]\n",
    "                    X[phase].append(z[node_indices].detach().cpu().numpy())\n",
    "\n",
    "                Y[phase].extend(batch.y.detach().cpu().numpy().flatten().tolist())\n",
    "                \n",
    "        # standardize the features based on mean and std of the training data\n",
    "        ss = StandardScaler()\n",
    "        ss.fit(np.concatenate(X['train'], 0))\n",
    "        for phase in phases:\n",
    "            for i in range(len(X[phase])):\n",
    "                X[phase][i] = ss.transform(X[phase][i])\n",
    "\n",
    "        # apply PCA if needed\n",
    "        if num_pca_components > 0:\n",
    "            print('Now running PCA ...')\n",
    "            pca = PCA(n_components=num_pca_components, random_state=random_seed)\n",
    "            pca.fit(np.concatenate(X['train'], 0))\n",
    "            for phase in phases:\n",
    "                for i in range(len(X[phase])):\n",
    "                    X[phase][i] = pca.transform(X[phase][i])\n",
    "                    \n",
    "            # plot the variance % explained by PCA components\n",
    "            plt.plot(np.arange(1, num_pca_components + 1), pca.explained_variance_ratio_, 'o--')\n",
    "            plt.grid(True)\n",
    "            plt.xlabel('Principal component')\n",
    "            plt.ylabel('Eigenvalue')\n",
    "            plt.xticks(np.arange(1, num_pca_components + 1, step=2))\n",
    "            plt.show()\n",
    "\n",
    "        # number of samples in the template distribution\n",
    "        N = int(round(np.asarray([x.shape[0] for x in X['train']]).mean()))\n",
    "\n",
    "        # derive the template distribution using K-means\n",
    "        print('Now running k-means for deriving the template ...')\n",
    "        kmeans = KMeans(n_clusters=N, verbose=verbose, random_state=random_seed)\n",
    "        kmeans.fit(np.concatenate(X['train'], 0))\n",
    "        template = kmeans.cluster_centers_\n",
    "\n",
    "        # calculate the final graph embeddings based on LOT\n",
    "        V = defaultdict(list)\n",
    "        for phase in phases:\n",
    "            print('Now deriving the final graph embeddings for the ' + phase + ' data ...')\n",
    "            for x in tqdm(X[phase]):\n",
    "                M = x.shape[0]\n",
    "                C = ot.dist(x, template)\n",
    "                b = np.ones((N,)) / float(N)\n",
    "                a = np.ones((M,)) / float(M)\n",
    "                p = ot.emd(a,b,C) # exact linear program\n",
    "                V[phase].append(np.matmul((N * p).T, x) - template)\n",
    "            V[phase] = np.stack(V[phase])\n",
    "\n",
    "        \n",
    "        # create the parameter grid for random forest\n",
    "        param_grid_RF = {\n",
    "            'max_depth': [None],\n",
    "            'min_samples_leaf': [1, 2, 5],\n",
    "            'min_samples_split': [2, 5, 10],\n",
    "            'n_estimators': [25, 50, 100, 150, 200]\n",
    "        }\n",
    "        \n",
    "        param_grid_all = {'RF': param_grid_RF}\n",
    "\n",
    "        # load the ROC-AUC evaluator\n",
    "        evaluator = Evaluator(name=dataset.name)\n",
    "        \n",
    "        # run the classifier\n",
    "        print('Now running the classifiers ...')\n",
    "        for classifier in classifiers:\n",
    "            if classifier not in param_grid_all:\n",
    "                print('Classifier {} not supported! Skipping ...'.format(classifier))\n",
    "                continue\n",
    "\n",
    "            param_grid = param_grid_all[classifier]\n",
    "\n",
    "            # determine train and validation index split for grid search\n",
    "            test_fold = [-1] * len(V['train']) + [0] * len(V['valid'])\n",
    "            ps = PredefinedSplit(test_fold)\n",
    "\n",
    "            # concatenate train and validation datasets\n",
    "            X_grid_search = np.concatenate((V['train'], V['valid']), axis=0)\n",
    "            X_grid_search = X_grid_search.reshape(X_grid_search.shape[0], -1)\n",
    "            Y_grid_search = np.concatenate((Y['train'], Y['valid']), axis=0)\n",
    "\n",
    "            results = defaultdict(list)\n",
    "            for experiment in range(num_experiments):\n",
    "                \n",
    "                # Create a base model\n",
    "                if classifier == 'RF':\n",
    "                    model = RandomForestClassifier(n_jobs=n_jobs, class_weight='balanced',\n",
    "                                                   random_state=random_seed+experiment)\n",
    "                \n",
    "                # Instantiate the grid search model\n",
    "                grid_search = GridSearchCV(estimator=model, param_grid=param_grid, \n",
    "                                          cv=ps, n_jobs=n_jobs, verbose=verbose, refit=False)\n",
    "\n",
    "                # Fit the grid search to the data\n",
    "                grid_search.fit(X_grid_search, Y_grid_search)\n",
    "\n",
    "                # Fit the model with best parameters on the training data (again)\n",
    "                for param in grid_search.best_params_:\n",
    "                    model.param = grid_search.best_params_[param]\n",
    "                model.fit(V['train'].reshape(V['train'].shape[0], -1), Y['train'])\n",
    "                \n",
    "                # Evaluate the performance\n",
    "                for phase in phases:\n",
    "                    pred_probs = model.predict_proba(V[phase].reshape(V[phase].shape[0], -1))\n",
    "                    input_dict = {'y_true': np.array(Y[phase]).reshape(-1,1),\n",
    "                                  'y_pred': pred_probs[:, 1].reshape(-1,1)}\n",
    "                    result_dict = evaluator.eval(input_dict)\n",
    "                    results[phase].append(result_dict['rocauc'])\n",
    "\n",
    "                print('experiment {0}/{1} for {2} completed ...'.format\\\n",
    "                      (experiment+1, num_experiments, classifier))\n",
    "            \n",
    "            results_table.add_row([classifier, str(L), str(F)] + ['{0:.2f} $\\pm$ {1:.2f}'.\\\n",
    "                                      format(100 * np.mean(results[phase]), \\\n",
    "                                             100 * np.std(results[phase])) for phase in phases])\n",
    "                  \n",
    "    print('\\n\\n' + results_table.title)\n",
    "    print(results_table)\n",
    "    return results_table"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Specify the parameters\n",
    "\n",
    "# num_hidden_layers = range(3, 9)\n",
    "num_hidden_layers = [4]\n",
    "\n",
    "# node_embedding_sizes = [100, 300, 500]\n",
    "node_embedding_sizes = [300]\n",
    "\n",
    "# final_node_embeddings = ['concat', 'avg', 'final']\n",
    "final_node_embeddings = ['final']\n",
    "\n",
    "num_pca_components = 20\n",
    "num_experiments = 10\n",
    "classifiers = ['RF']\n",
    "device = 'cpu'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "****************************************************************************************************\n",
      "# diffusion layers = 4, node embedding size = 300, node embedding mode: final\n",
      "Now diffusing the train data ...\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "7d4ccd228d9e40b1a9e92c5f68aca84f",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "HBox(children=(FloatProgress(value=0.0, max=1029.0), HTML(value='')))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Now diffusing the valid data ...\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "c3980cfe626041f08e05b68ae256666e",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "HBox(children=(FloatProgress(value=0.0, max=129.0), HTML(value='')))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Now diffusing the test data ...\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "8c9f088923ea4e78af34b98b8c99250d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "HBox(children=(FloatProgress(value=0.0, max=129.0), HTML(value='')))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Now running PCA ...\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Now running k-means for deriving the template ...\n",
      "Now deriving the final graph embeddings for the train data ...\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "d4b63ae5f05747448cee28917d2ee563",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "HBox(children=(FloatProgress(value=0.0, max=32901.0), HTML(value='')))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Now deriving the final graph embeddings for the valid data ...\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "e8eff76a60e047e88cce0d605acbe186",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "HBox(children=(FloatProgress(value=0.0, max=4113.0), HTML(value='')))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Now deriving the final graph embeddings for the test data ...\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "8754d35634f64e9b85344095771c029d",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "HBox(children=(FloatProgress(value=0.0, max=4113.0), HTML(value='')))"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Now running the classifiers ...\n",
      "experiment 1/10 for RF completed ...\n",
      "experiment 2/10 for RF completed ...\n",
      "experiment 3/10 for RF completed ...\n",
      "experiment 4/10 for RF completed ...\n",
      "experiment 6/10 for RF completed ...\n",
      "experiment 7/10 for RF completed ...\n",
      "experiment 8/10 for RF completed ...\n",
      "experiment 9/10 for RF completed ...\n",
      "experiment 10/10 for RF completed ...\n",
      "\n",
      "\n",
      "Final ROC-AUC(%) results for the ogbg-molhiv dataset with 'final' node embedding and one-hot 13-dim edge embedding\n",
      "+------------+--------------------+---------------------+-------------------+------------------+------------------+\n",
      "| Classifier | # Diffusion Layers | Node Embedding Size |       Train.      |       Val.       |       Test       |\n",
      "+------------+--------------------+---------------------+-------------------+------------------+------------------+\n",
      "|     RF     |         4          |         300         | 100.00 $\\pm$ 0.00 | 80.37 $\\pm$ 1.74 | 77.56 $\\pm$ 1.60 |\n",
      "+------------+--------------------+---------------------+-------------------+------------------+------------------+\n"
     ]
    }
   ],
   "source": [
    "# Run the algorithm\n",
    "for final_node_embedding in final_node_embeddings:\n",
    "    WEGL(dataset=dataset,\n",
    "         num_hidden_layers=num_hidden_layers,\n",
    "         node_embedding_sizes=node_embedding_sizes,\n",
    "         final_node_embedding=final_node_embedding,\n",
    "         num_pca_components=num_pca_components,\n",
    "         num_experiments=num_experiments,\n",
    "         classifiers=classifiers,\n",
    "         device=device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "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.6.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
