"vscode:/vscode.git/clone" did not exist on "b540e5d1b95bfd3a0380a8b4b7c6e6d6bdb07ef1"
train.py 4.39 KB
Newer Older
1
2
import argparse
import time
3
import numpy as np
4
import networkx as nx
5
6
7
import torch
import torch.nn as nn
import torch.nn.functional as F
8
9
10
import dgl
from dgl.data import register_data_args
from dgl.data import CoraGraphDataset, CiteseerGraphDataset, PubmedGraphDataset
11
12
13
14
15

from gcn import GCN
#from gcn_mp import GCN
#from gcn_spmv import GCN

16

17
18
19
20
21
22
23
24
25
26
def evaluate(model, features, labels, mask):
    model.eval()
    with torch.no_grad():
        logits = model(features)
        logits = logits[mask]
        labels = labels[mask]
        _, indices = torch.max(logits, dim=1)
        correct = torch.sum(indices == labels)
        return correct.item() * 1.0 / len(labels)

27

28
29
def main(args):
    # load and preprocess dataset
30
31
32
33
34
35
    if args.dataset == 'cora':
        data = CoraGraphDataset()
    elif args.dataset == 'citeseer':
        data = CiteseerGraphDataset()
    elif args.dataset == 'pubmed':
        data = PubmedGraphDataset()
36
    else:
37
38
39
40
41
42
43
        raise ValueError('Unknown dataset: {}'.format(args.dataset))

    g = data[0]
    if args.gpu < 0:
        cuda = False
    else:
        cuda = True
44
        g = g.int().to(args.gpu)
45
46
47
48
49
50

    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']
51
52
53
54
55
56
57
58
59
60
    in_feats = features.shape[1]
    n_classes = data.num_labels
    n_edges = data.graph.number_of_edges()
    print("""----Data statistics------'
      #Edges %d
      #Classes %d
      #Train samples %d
      #Val samples %d
      #Test samples %d""" %
          (n_edges, n_classes,
Zihao Ye's avatar
Zihao Ye committed
61
62
63
              train_mask.int().sum().item(),
              val_mask.int().sum().item(),
              test_mask.int().sum().item()))
64
65

    # add self loop
Da Zheng's avatar
Da Zheng committed
66
    if args.self_loop:
67
68
        g = dgl.remove_self_loop(g)
        g = dgl.add_self_loop(g)
69
    n_edges = g.number_of_edges()
70

71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
    # normalization
    degs = g.in_degrees().float()
    norm = torch.pow(degs, -0.5)
    norm[torch.isinf(norm)] = 0
    if cuda:
        norm = norm.cuda()
    g.ndata['norm'] = norm.unsqueeze(1)

    # create GCN model
    model = GCN(g,
                in_feats,
                args.n_hidden,
                n_classes,
                args.n_layers,
                F.relu,
                args.dropout)

    if cuda:
        model.cuda()
    loss_fcn = torch.nn.CrossEntropyLoss()

    # use optimizer
    optimizer = torch.optim.Adam(model.parameters(),
                                 lr=args.lr,
                                 weight_decay=args.weight_decay)

    # initialize graph
    dur = []
    for epoch in range(args.n_epochs):
        model.train()
        if epoch >= 3:
            t0 = time.time()
        # forward
        logits = model(features)
        loss = loss_fcn(logits[train_mask], labels[train_mask])

        optimizer.zero_grad()
        loss.backward()
        optimizer.step()

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

        acc = evaluate(model, features, labels, val_mask)
        print("Epoch {:05d} | Time(s) {:.4f} | Loss {:.4f} | Accuracy {:.4f} | "
              "ETputs(KTEPS) {:.2f}". format(epoch, np.mean(dur), loss.item(),
                                             acc, n_edges / np.mean(dur) / 1000))

    print()
    acc = evaluate(model, features, labels, test_mask)
121
    print("Test accuracy {:.2%}".format(acc))
122
123
124
125
126
127


if __name__ == '__main__':
    parser = argparse.ArgumentParser(description='GCN')
    register_data_args(parser)
    parser.add_argument("--dropout", type=float, default=0.5,
128
                        help="dropout probability")
129
    parser.add_argument("--gpu", type=int, default=-1,
130
                        help="gpu")
131
    parser.add_argument("--lr", type=float, default=1e-2,
132
                        help="learning rate")
133
    parser.add_argument("--n-epochs", type=int, default=200,
134
                        help="number of training epochs")
135
    parser.add_argument("--n-hidden", type=int, default=16,
136
                        help="number of hidden gcn units")
137
    parser.add_argument("--n-layers", type=int, default=1,
138
                        help="number of hidden gcn layers")
139
    parser.add_argument("--weight-decay", type=float, default=5e-4,
140
                        help="Weight for L2 loss")
Da Zheng's avatar
Da Zheng committed
141
    parser.add_argument("--self-loop", action='store_true',
142
                        help="graph self-loop (default=False)")
Da Zheng's avatar
Da Zheng committed
143
    parser.set_defaults(self_loop=False)
144
145
146
147
    args = parser.parse_args()
    print(args)

    main(args)