forked from Wo1v3r/trollaganda
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathembedding.py
188 lines (151 loc) · 6.73 KB
/
embedding.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
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
"""
This module contains the text pre-process, and embed it.
"""
import os
import zipfile
import gensim
import matplotlib.pyplot as plt
import numpy as np
import pandas as pd
import seaborn as sns
from keras.preprocessing.sequence import pad_sequences
from keras.preprocessing.text import Tokenizer
from nltk.tokenize import RegexpTokenizer
from sklearn.model_selection import train_test_split
from sklearn.preprocessing import LabelEncoder
import config
class PreProcess(object):
"""
This class will pre-process pandas data frame
"""
def __init__(self, data, textfield):
self.data = data
self.textfield = textfield
def process_text(self):
self.data[self.textfield] = self.data[self.textfield].str.replace(r"http\S+", "LINK")
self.data[self.textfield] = self.data[self.textfield].str.replace(r"@\S+", "TAG")
self.data[self.textfield] = self.data[self.textfield].str.replace(r"[^A-Za-z0-9(),!?@\'\`\"\_\n]", " ")
self.data[self.textfield] = self.data[self.textfield].str.replace(r"@", "AT")
self.data[self.textfield] = self.data[self.textfield].str.lower()
return self.data
def see_data_head(self):
self.data.head()
class PrepareEmbedding(object):
"""
This class is used to create the embedding on the data
"""
def __init__(self, X, Y, embedded_path, test_size=0.15):
self.X = X
self.Y = Y
self.test_size = test_size
self.embedded_path = embedded_path
self.pre_train = None
self._prepare_labels()
self.X_train, self.X_test, self.Y_train, self.Y_test = train_test_split(self.X, self.Y, test_size=self.test_size)
self.train_embedding_weights = None
self.train_cnn_data = None
self.test_cnn_data = None
self.train_word_index = None
def _prepare_labels(self):
le = LabelEncoder()
self.Y = le.fit_transform(self.Y)
self.Y = self.Y.reshape(-1, 1)
def _tokenize_messages(self):
# TOKENIZING THE TEXT
regextokenizer = RegexpTokenizer(r"\w+")
self.X_train = self.X_train.apply(regextokenizer.tokenize)
# delete Stop Words
self.X_train = self.X_train.apply(lambda vec: [word for word in vec if word not in config.stopwords0])
return self.X_train
def print_info(self):
train_tokens = self._tokenize_messages()
all_training_words = [word for tokens in train_tokens for word in tokens]
training_sentence_lengths = [len(tokens) for tokens in train_tokens]
training_vocab = sorted(list(set(all_training_words)))
print("Total: %s words, vocabulary size of %s" % (len(all_training_words), len(training_vocab)))
print("Max sentence length is %s" % max(training_sentence_lengths))
def load_word_2_vec(self):
print("Loading W2V")
self.pre_train = gensim.models.KeyedVectors.load_word2vec_format(self.embedded_path, binary=True)
print("W2V Loaded")
def load_glove(self):
print("Loading GloVe")
self.pre_train = {}
# https: // nlp.stanford.edu / projects / glove /
if not os.path.isfile(self.embedded_path): # should be .txt
zip_ref = zipfile.ZipFile(self.embedded_path, 'r') # should be .zip
zip_ref.extractall('./')
zip_ref.close()
f = open(self.embedded_path)
for line in f:
values = line.split(" ")
word = values[0]
coefs = np.asarray(values[1:], dtype="float32")
self.pre_train[word] = coefs
f.close()
print("GloVe data loaded")
def train(self, max_vocab_size=config.MAXVOCABSIZE):
try:
if not self.pre_train:
raise Exception("Pre trained vocabulary isn't loaded.")
except Exception as e:
print(e)
return
tokenizer = Tokenizer(num_words=max_vocab_size, lower=True, char_level=False)
tokenizer.fit_on_texts(self.X_train)
# TODO: Should save tokenizer along with model incase of predicting with a loaded model
self.tokenizer = tokenizer
training_sequences = tokenizer.texts_to_sequences(self.X_train)
self.train_word_index = tokenizer.word_index
self.train_cnn_data = pad_sequences(training_sequences, maxlen=config.MAXSEQLENGTH)
self.train_embedding_weights = np.zeros((len(self.train_word_index) + 1, config.EMBEDDINGDIM))
for word, index in self.train_word_index.items():
self.train_embedding_weights[index, :] = self.pre_train[word] if word in self.pre_train \
else np.random.rand(config.EMBEDDINGDIM)
# Prepare the test data
test_sequences = tokenizer.texts_to_sequences(self.X_test)
self.test_cnn_data = pad_sequences(test_sequences, maxlen=config.MAXSEQLENGTH)
print("Found {} unique tokens.".format(len(self.train_word_index)))
def release_pre_trained(self):
del self.pre_train
self.pre_train = None
def preprocess_predictions(self, messages):
# def _standardize_text(self, text):
# text = text.str.replace(r"http\S+", "LINK")
# text = text.str.replace(r"@\S+", "TAG")
# text = text.str.replace(r"[^A-Za-z0-9(),!?@\'\`\"\_\n]", " ")
# text = text.str.replace(r"@", "AT ")
# text = text.str.lower()
# return text
def _tokenize_data(message_series):
# TOKENIZING THE TEXT
regextokenizer = RegexpTokenizer(r"\w+")
messages = message_series.apply(regextokenizer.tokenize)
# delete Stop Words
messages = messages.apply(lambda vec: [word for word in vec if word not in config.stopwords0])
return messages
series = pd.Series(messages)
series = series.astype(str)
tokenized_data = _tokenize_data(message_series=series)
predictSequences = self.tokenizer.texts_to_sequences(tokenized_data)
predictData = pad_sequences(predictSequences, maxlen=config.MAXSEQLENGTH)
return predictData
class ReadFile(object):
"""
Reading the datasets class.
The data set should be two columns,
tweet and classification(1 is troll)
"""
def __init__(self, path, split=None):
self.path = path
self.split = split
self.data = None
def readfile(self):
self.data = pd.read_csv(self.path, delimiter=",", encoding="utf8", names=["message", "isTroll"])
self.data.message = self.data.message.astype(str)
self.data = self.data.iloc[:self.split] if self.split > 0 else self.data
def distribution_plot(self):
if self.data is not None:
sns.countplot(self.data.isTroll)
plt.xlabel("Label")
plt.title("Number of troll or not messages")