-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpredict.py
145 lines (132 loc) · 5.8 KB
/
predict.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
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
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
#!/usr/bin/env python
# encoding: utf-8
# File Name: predict.py
# Author: Jiezhong Qiu
# Create Time: 2017/07/17 21:57
# TODO:
import os
import pickle as pkl
import numpy as np
import scipy.io
import argparse
import logging
import sys
import scipy.sparse as sparse
from sklearn.linear_model import LogisticRegression
from sklearn.model_selection import ShuffleSplit
from sklearn.multiclass import OneVsRestClassifier
from sklearn.metrics import f1_score
from sklearn.exceptions import UndefinedMetricWarning
import warnings
warnings.filterwarnings("ignore", category=UserWarning)
warnings.filterwarnings("ignore", category=UndefinedMetricWarning)
logger = logging.getLogger(__name__)
def construct_indicator(y_score, y):
# rank the labels by the scores directly
num_label = np.sum(y, axis=1, dtype=np.int)
y_sort = np.fliplr(np.argsort(y_score, axis=1))
y_pred = np.zeros_like(y, dtype=np.int)
for i in range(y.shape[0]):
for j in range(num_label[i]):
y_pred[i, y_sort[i, j]] = 1
return y_pred
def load_w2v_feature(file):
with open(file, "rb") as f:
nu = 0
for line in f:
content = line.strip().split()
nu += 1
if nu == 1:
n, d = int(content[0]), int(content[1])
feature = [[] for i in range(n)]
continue
index = int(content[0])
for x in content[1:]:
feature[index].append(float(x))
for item in feature:
assert len(item) == d
return np.array(feature, dtype=np.float32)
def load_label(file, variable_name="group"):
data = scipy.io.loadmat(file)
logger.info("loading mat file %s", file)
label = data[variable_name].todense().astype(np.int)
label = np.array(label)
print(label.shape, type(label), label.min(), label.max())
return label
def predict_cv(X, y, train_ratio=0.2, n_splits=10, random_state=0, C=1.):
micro, macro = [], []
shuffle = ShuffleSplit(n_splits=n_splits, test_size=1-train_ratio,
random_state=random_state)
for train_index, test_index in shuffle.split(X):
print(train_index.shape, test_index.shape)
assert len(set(train_index) & set(test_index)) == 0
assert len(train_index) + len(test_index) == X.shape[0]
X_train, X_test = X[train_index], X[test_index]
y_train, y_test = y[train_index], y[test_index]
clf = OneVsRestClassifier(
LogisticRegression(
C=C,
solver="liblinear",
multi_class="ovr"),
n_jobs=-1)
clf.fit(X_train, y_train)
y_score = clf.predict_proba(X_test)
y_pred = construct_indicator(y_score, y_test)
mi = f1_score(y_test, y_pred, average="micro")
ma = f1_score(y_test, y_pred, average="macro")
logger.info("micro f1 %f macro f1 %f", mi, ma)
micro.append(mi)
macro.append(ma)
logger.info("%d fold validation, training ratio %f", len(micro), train_ratio)
logger.info("Average micro %.2f, Average macro %.2f",
np.mean(micro) * 100,
np.mean(macro) * 100)
if __name__ == "__main__":
parser = argparse.ArgumentParser()
parser.add_argument("--label", type=str, required=True,
help="input file path for labels (.mat)")
parser.add_argument("--embedding", type=str, required=True,
help="input file path for embedding (.npy)")
parser.add_argument("--matfile-variable-name", type=str, default='group',
help='variable name of adjacency matrix inside a .mat file.')
parser.add_argument("--seed", type=int, required=True,
help="seed used for random number generator when randomly split data into training/test set.")
parser.add_argument("--start-train-ratio", type=int, default=10,
help="the start value of the train ratio (inclusive).")
parser.add_argument("--stop-train-ratio", type=int, default=90,
help="the end value of the train ratio (inclusive).")
parser.add_argument("--num-train-ratio", type=int, default=9,
help="the number of train ratio choosed from [train-ratio-start, train-ratio-end].")
parser.add_argument("--C", type=float, default=1.0,
help="inverse of regularization strength used in logistic regression.")
parser.add_argument("--num-split", type=int, default=10,
help="The number of re-shuffling & splitting for each train ratio.")
parser.add_argument("--feat-norm", action='store_true',
help="Whether normalizing feature vector, such that each vector has unit norm")
args = parser.parse_args()
logging.basicConfig(
stream=sys.stdout,
#filename="%s.log" % args.embedding, filemode="w", # uncomment this to log to file
level=logging.INFO,
format='%(asctime)s %(message)s') # include timestamp
logger.info("Loading label from %s...", args.label)
label = load_label(file=args.label, variable_name=args.matfile_variable_name)
logger.info("Label loaded!")
logger.info("Loading network embedding from %s...", args.embedding)
ext = os.path.splitext(args.embedding)[1]
if ext == ".npy":
embedding = np.load(args.embedding)
elif ext == ".pkl":
with open(args.embedding, "rb") as f:
embedding = pkl.load(f)
else:
# Load word2vec format
embedding = load_w2v_feature(args.embedding)
if args.feat_norm:
embedding = sparse.diags(1 / np.sqrt(np.power(embedding, 2).sum(axis=1))).dot(embedding)
logger.info("Network embedding loaded!")
train_ratios = np.linspace(args.start_train_ratio, args.stop_train_ratio,
args.num_train_ratio)
for tr in train_ratios:
predict_cv(embedding, label, train_ratio=tr/100.,
n_splits=args.num_split, C=args.C, random_state=args.seed)