import torch.nn as nn
import torch.nn.functional as F
from torch_geometric.nn import GCNConv, global_add_pool

class GCN(nn.Module):
    def __init__(self, num_features, num_hidden, num_classes, num_layers, graph_class):
        super(GCN, self).__init__()

        num_features = num_features
        dim = num_hidden

        self.num_layers = num_layers
        self.graph_class = graph_class

        self.convs = nn.ModuleList()
        self.bns = nn.ModuleList()
        self.fcs = nn.ModuleList()

        self.convs.append(GCNConv(in_channels=num_features, out_channels=dim, improved=True))
        self.bns.append(nn.BatchNorm1d(dim))
        self.fcs.append(nn.Linear(num_features, num_classes))
        self.fcs.append(nn.Linear(dim, num_classes))

        for i in range(self.num_layers - 1):
            self.convs.append(GCNConv(dim, dim, improved=True))
            self.bns.append(nn.BatchNorm1d(dim))
            self.fcs.append(nn.Linear(dim, num_classes))

    def reset_parameters(self):
        for m in self.modules():
            if isinstance(m, nn.Linear):
                m.reset_parameters()
            elif isinstance(m, GCNConv):
                m.reset_parameters()
            elif isinstance(m, nn.BatchNorm1d):
                m.reset_parameters()

    def forward(self, data):
        x = data.x
        edge_index = data.edge_index
        batch = data.batch

        outs = [x]
        for i in range(self.num_layers):
            x = self.convs[i](x, edge_index)
            x = self.bns[i](x)
            x = F.relu(x)
            outs.append(x)

        out = None
        for i, x in enumerate(outs):
            if self.graph_class:
                x = global_add_pool(x, batch)
            x = self.fcs[i](x)  # No dropout for these experiments
            if out is None:
                out = x
            else:
                out += x
        return F.log_softmax(out, dim=-1), 0