-
Notifications
You must be signed in to change notification settings - Fork 33
/
Copy pathdata_helpers.py
executable file
·113 lines (96 loc) · 4.32 KB
/
data_helpers.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
import numpy as np
import os
import re
import itertools
import scipy.sparse as sp
import cPickle as pickle
from collections import Counter
from nltk.corpus import stopwords
cachedStopWords = stopwords.words("english")
def clean_str(string):
# remove stopwords
# string = ' '.join([word for word in string.split() if word not in cachedStopWords])
string = re.sub(r"[^A-Za-z0-9(),!?\'\`]", " ", string)
string = re.sub(r"\'s", " \'s", string)
string = re.sub(r"\'ve", " \'ve", string)
string = re.sub(r"n\'t", " n\'t", string)
string = re.sub(r"\'re", " \'re", string)
string = re.sub(r"\'d", " \'d", string)
string = re.sub(r"\'ll", " \'ll", string)
string = re.sub(r",", " , ", string)
string = re.sub(r"!", " ! ", string)
string = re.sub(r"\(", " \( ", string)
string = re.sub(r"\)", " \) ", string)
string = re.sub(r"\?", " \? ", string)
string = re.sub(r"\s{2,}", " ", string)
return string.strip().lower()
def pad_sentences(sentences, padding_word="<PAD/>", max_length=500):
sequence_length = min(max(len(x) for x in sentences), max_length)
padded_sentences = []
for i in range(len(sentences)):
sentence = sentences[i]
if len(sentence) < max_length:
num_padding = sequence_length - len(sentence)
new_sentence = sentence + [padding_word] * num_padding
else:
new_sentence = sentence[:max_length]
padded_sentences.append(new_sentence)
return padded_sentences
def load_data_and_labels(data):
x_text = [clean_str(doc['text']) for doc in data]
x_text = [s.split(" ") for s in x_text]
labels = [doc['catgy'] for doc in data]
row_idx, col_idx, val_idx = [], [], []
for i in range(len(labels)):
l_list = list(set(labels[i])) # remove duplicate cateories to avoid double count
for y in l_list:
row_idx.append(i)
col_idx.append(y)
val_idx.append(1)
m = max(row_idx) + 1
n = max(col_idx) + 1
Y = sp.csr_matrix((val_idx, (row_idx, col_idx)), shape=(m, n))
return [x_text, Y, labels]
def build_vocab(sentences, vocab_size=50000):
word_counts = Counter(itertools.chain(*sentences))
vocabulary_inv = [x[0] for x in word_counts.most_common(vocab_size)]
vocabulary = {x: i for i, x in enumerate(vocabulary_inv)}
# append <UNK/> symbol to the vocabulary
vocabulary['<UNK/>'] = len(vocabulary)
vocabulary_inv.append('<UNK/>')
return [vocabulary, vocabulary_inv]
def build_input_data(sentences, vocabulary):
x = np.array([[vocabulary[word] if word in vocabulary else vocabulary['<UNK/>'] for word in sentence] for sentence in sentences])
#x = np.array([[vocabulary[word] if word in vocabulary else len(vocabulary) for word in sentence] for sentence in sentences])
return x
def load_data(data_path, max_length=500, vocab_size=50000):
# Load and preprocess data
with open(os.path.join(data_path), 'rb') as fin:
[train, test, vocab, catgy] = pickle.load(fin)
# dirty trick to prevent errors happen when test is empty
if len(test) == 0:
test[:5] = train[:5]
trn_sents, Y_trn, Y_trn_o = load_data_and_labels(train)
tst_sents, Y_tst, Y_tst_o = load_data_and_labels(test)
trn_sents_padded = pad_sentences(trn_sents, max_length=max_length)
tst_sents_padded = pad_sentences(tst_sents, max_length=max_length)
vocabulary, vocabulary_inv = build_vocab(trn_sents_padded + tst_sents_padded, vocab_size=vocab_size)
X_trn = build_input_data(trn_sents_padded, vocabulary)
X_tst = build_input_data(tst_sents_padded, vocabulary)
return X_trn, Y_trn, Y_trn_o, X_tst, Y_tst, Y_tst_o, vocabulary, vocabulary_inv
# return X_trn, Y_trn, vocabulary, vocabulary_inv
def batch_iter(data, batch_size, num_epochs):
"""
Generates a batch iterator for a dataset.
"""
data = np.array(data)
data_size = len(data)
num_batches_per_epoch = int(len(data)/batch_size) + 1
for epoch in range(num_epochs):
# Shuffle the data at each epoch
shuffle_indices = np.random.permutation(np.arange(data_size))
shuffled_data = data[shuffle_indices]
for batch_num in range(num_batches_per_epoch):
start_index = batch_num * batch_size
end_index = min((batch_num + 1) * batch_size, data_size)
yield shuffled_data[start_index:end_index]