gcn_builtin.py 6.91 KB
Newer Older
VoVAllen's avatar
VoVAllen committed
1
2
import argparse
import math
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
3
4
5
6
import time

import dgl
import dgl.function as fn
VoVAllen's avatar
VoVAllen committed
7
import networkx as nx
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
8
import numpy as np
VoVAllen's avatar
VoVAllen committed
9
import tensorflow as tf
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
10
11
12
13
14
15
from dgl.data import (
    CiteseerGraphDataset,
    CoraGraphDataset,
    PubmedGraphDataset,
    register_data_args,
)
VoVAllen's avatar
VoVAllen committed
16
17
18
19
from tensorflow.keras import layers


class GCNLayer(layers.Layer):
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
20
    def __init__(self, g, in_feats, out_feats, activation, dropout, bias=True):
VoVAllen's avatar
VoVAllen committed
21
22
23
        super(GCNLayer, self).__init__()
        self.g = g

24
        w_init = tf.keras.initializers.VarianceScaling(
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
25
26
27
28
29
30
            scale=1.0, mode="fan_out", distribution="uniform"
        )
        self.weight = tf.Variable(
            initial_value=w_init(shape=(in_feats, out_feats), dtype="float32"),
            trainable=True,
        )
VoVAllen's avatar
VoVAllen committed
31
32
33
        if dropout:
            self.dropout = layers.Dropout(rate=dropout)
        else:
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
34
            self.dropout = 0.0
VoVAllen's avatar
VoVAllen committed
35
36
        if bias:
            b_init = tf.zeros_initializer()
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
37
38
39
40
            self.bias = tf.Variable(
                initial_value=b_init(shape=(out_feats,), dtype="float32"),
                trainable=True,
            )
VoVAllen's avatar
VoVAllen committed
41
42
43
44
45
46
47
        else:
            self.bias = None
        self.activation = activation

    def call(self, h):
        if self.dropout:
            h = self.dropout(h)
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
48
49
50
51
        self.g.ndata["h"] = tf.matmul(h, self.weight)
        self.g.ndata["norm_h"] = self.g.ndata["h"] * self.g.ndata["norm"]
        self.g.update_all(fn.copy_u("norm_h", "m"), fn.sum("m", "h"))
        h = self.g.ndata["h"]
VoVAllen's avatar
VoVAllen committed
52
53
54
55
56
57
58
59
        if self.bias is not None:
            h = h + self.bias
        if self.activation:
            h = self.activation(h)
        return h


class GCN(layers.Layer):
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
60
61
62
    def __init__(
        self, g, in_feats, n_hidden, n_classes, n_layers, activation, dropout
    ):
VoVAllen's avatar
VoVAllen committed
63
64
65
66
        super(GCN, self).__init__()
        self.layers = []

        # input layer
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
67
        self.layers.append(GCNLayer(g, in_feats, n_hidden, activation, dropout))
VoVAllen's avatar
VoVAllen committed
68
69
70
        # hidden layers
        for i in range(n_layers - 1):
            self.layers.append(
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
71
72
                GCNLayer(g, n_hidden, n_hidden, activation, dropout)
            )
VoVAllen's avatar
VoVAllen committed
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
        # output layer
        self.layers.append(GCNLayer(g, n_hidden, n_classes, None, dropout))

    def call(self, features):
        h = features
        for layer in self.layers:
            h = layer(h)
        return h


def evaluate(model, features, labels, mask):
    logits = model(features, training=False)
    logits = logits[mask]
    labels = labels[mask]
    indices = tf.math.argmax(logits, axis=1)
    acc = tf.reduce_mean(tf.cast(indices == labels, dtype=tf.float32))
    return acc.numpy().item()


def main(args):
    # load and preprocess dataset
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
94
    if args.dataset == "cora":
95
        data = CoraGraphDataset()
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
96
    elif args.dataset == "citeseer":
97
        data = CiteseerGraphDataset()
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
98
    elif args.dataset == "pubmed":
99
100
        data = PubmedGraphDataset()
    else:
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
101
        raise ValueError("Unknown dataset: {}".format(args.dataset))
VoVAllen's avatar
VoVAllen committed
102

103
    g = data[0]
VoVAllen's avatar
VoVAllen committed
104
105
106
107
    if args.gpu < 0:
        device = "/cpu:0"
    else:
        device = "/gpu:{}".format(args.gpu)
108
        g = g.to(device)
VoVAllen's avatar
VoVAllen committed
109
110

    with tf.device(device):
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
111
112
113
114
115
        features = g.ndata["feat"]
        labels = g.ndata["label"]
        train_mask = g.ndata["train_mask"]
        val_mask = g.ndata["val_mask"]
        test_mask = g.ndata["test_mask"]
VoVAllen's avatar
VoVAllen committed
116
117
118
        in_feats = features.shape[1]
        n_classes = data.num_labels
        n_edges = data.graph.number_of_edges()
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
119
120
        print(
            """----Data statistics------'
VoVAllen's avatar
VoVAllen committed
121
122
123
124
        #Edges %d
        #Classes %d
        #Train samples %d
        #Val samples %d
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
125
126
127
128
129
130
131
132
133
        #Test samples %d"""
            % (
                n_edges,
                n_classes,
                train_mask.numpy().sum(),
                val_mask.numpy().sum(),
                test_mask.numpy().sum(),
            )
        )
VoVAllen's avatar
VoVAllen committed
134

135
136
137
        # add self loop
        g = dgl.remove_self_loop(g)
        g = dgl.add_self_loop(g)
VoVAllen's avatar
VoVAllen committed
138
139
140
141
142
143
        n_edges = g.number_of_edges()
        # # normalization
        degs = tf.cast(tf.identity(g.in_degrees()), dtype=tf.float32)
        norm = tf.math.pow(degs, -0.5)
        norm = tf.where(tf.math.is_inf(norm), tf.zeros_like(norm), norm)

Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
144
        g.ndata["norm"] = tf.expand_dims(norm, -1)
VoVAllen's avatar
VoVAllen committed
145
146

        # create GCN model
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
147
148
149
150
151
152
153
154
155
156
157
        model = GCN(
            g,
            in_feats,
            args.n_hidden,
            n_classes,
            args.n_layers,
            tf.nn.relu,
            args.dropout,
        )

        optimizer = tf.keras.optimizers.Adam(learning_rate=args.lr)
VoVAllen's avatar
VoVAllen committed
158
159

        loss_fcn = tf.keras.losses.SparseCategoricalCrossentropy(
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
160
161
            from_logits=True
        )
VoVAllen's avatar
VoVAllen committed
162
163
164
165
166
167
168
169
170
        # initialize graph
        dur = []
        for epoch in range(args.n_epochs):
            if epoch >= 3:
                t0 = time.time()
            # forward
            with tf.GradientTape() as tape:
                logits = model(features)
                loss_value = loss_fcn(labels[train_mask], logits[train_mask])
171
                # Manually Weight Decay
172
                # We found Tensorflow has a different implementation on weight decay
173
174
175
                # of Adam(W) optimizer with PyTorch. And this results in worse results.
                # Manually adding weights to the loss to do weight decay solves this problem.
                for weight in model.trainable_weights:
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
176
177
178
                    loss_value = loss_value + args.weight_decay * tf.nn.l2_loss(
                        weight
                    )
VoVAllen's avatar
VoVAllen committed
179
180
181
182
183
184
185
186

                grads = tape.gradient(loss_value, model.trainable_weights)
                optimizer.apply_gradients(zip(grads, model.trainable_weights))

            if epoch >= 3:
                dur.append(time.time() - t0)

            acc = evaluate(model, features, labels, val_mask)
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
187
188
189
190
191
192
193
194
195
196
            print(
                "Epoch {:05d} | Time(s) {:.4f} | Loss {:.4f} | Accuracy {:.4f} | "
                "ETputs(KTEPS) {:.2f}".format(
                    epoch,
                    np.mean(dur),
                    loss_value.numpy().item(),
                    acc,
                    n_edges / np.mean(dur) / 1000,
                )
            )
VoVAllen's avatar
VoVAllen committed
197
198
199
200
201

        acc = evaluate(model, features, labels, test_mask)
        print("Test Accuracy {:.4f}".format(acc))


Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
202
203
if __name__ == "__main__":
    parser = argparse.ArgumentParser(description="GCN")
VoVAllen's avatar
VoVAllen committed
204
    register_data_args(parser)
Hongzhi (Steve), Chen's avatar
Hongzhi (Steve), Chen committed
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
    parser.add_argument(
        "--dropout", type=float, default=0.5, help="dropout probability"
    )
    parser.add_argument("--gpu", type=int, default=-1, help="gpu")
    parser.add_argument("--lr", type=float, default=1e-2, help="learning rate")
    parser.add_argument(
        "--n-epochs", type=int, default=200, help="number of training epochs"
    )
    parser.add_argument(
        "--n-hidden", type=int, default=16, help="number of hidden gcn units"
    )
    parser.add_argument(
        "--n-layers", type=int, default=1, help="number of hidden gcn layers"
    )
    parser.add_argument(
        "--weight-decay", type=float, default=5e-4, help="Weight for L2 loss"
    )
VoVAllen's avatar
VoVAllen committed
222
223
224
225
    args = parser.parse_args()
    print(args)

    main(args)