-
Notifications
You must be signed in to change notification settings - Fork 33
/
Copy pathdata_init.py
218 lines (191 loc) · 9.15 KB
/
data_init.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
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
import os
import yaml
import shutil
import sidekit
import numpy as np
from tqdm import tqdm
from utils import convert_wav, safe_makedir, parse_yaml
class Initializer():
"""
To build speaker verification model, one needs speech data from each speaker
that is to be known by the system. The set of known speakers are in speaker
recognition known as the (enrollment speakers), and a speaker is enrolled
into the system when enrollment data from the speaker is processed to build
its model.
After the enrollment process, the performance of the speaker verification
system can be evaluated using test data, which in an open set scenario, will
consist of data from speakers in and outside the enrollment set.
The set of all speakers involved in testing the system will be referred to
as the test speakers.
This class if for preprocessing and structure the preprocessed data
into h5 files that will be used later for training and evaluating our models
NOTE:All outputs of this script can be found in the directory self.task_dir
"""
def __init__(self, conf_path):
"""
This method parses the YAML configuration file which can be used for
initializing the member varaibles!!
Args:
conf_path (String): path of the YAML configuration file
"""
#location of output files
self.conf = parse_yaml(conf_path)
self.task_dir = os.path.join(self.conf['outpath'], "task")
#location of audio files
self.audio_dir = os.path.join(self.conf['outpath'], "audio")
#location of all the audio data
self.data_dir = os.path.join(self.audio_dir, "data")
#location of just the enrollment audio data
self.enroll_dir = os.path.join(self.audio_dir, "enroll")
#location of just the test audio data
self.test_dir = os.path.join(self.audio_dir, "test")
def preprocess_audio(self):
"""
Copy the Merged Arabic Corpus of Isolated Words into their
associated directory. The whole audio data will be in 'data'
directory, the enrolled data only will be in 'enroll', and the
test data will be in 'test'.
"""
#remove the data directory if exists
if os.path.exists(self.data_dir):
shutil.rmtree(self.data_dir)
#iterate over speakers
speakers = sorted(os.listdir(self.conf['inpath']))
for sp in tqdm(speakers, desc="Converting Audio"):
speaker_path = os.path.join(self.conf['inpath'], sp)
wav_filenames = os.listdir(speaker_path)
for wav in wav_filenames:
inwav = os.path.join(speaker_path, wav)
outwav = os.path.join(self.data_dir, wav)
convert_wav(inwav,
outwav,
no_channels = self.conf['no_channels'],
sampling_rate = self.conf['sampling_rate'],
bit_precision = self.conf['bit_precision'])
#remove the enroll directory if exists
if os.path.exists(self.enroll_dir):
shutil.rmtree(self.enroll_dir)
#remove the test directory if exists
if os.path.exists(self.test_dir):
shutil.rmtree(self.test_dir)
#create audio/enroll directory
safe_makedir(self.enroll_dir)
#create audio/test directory
safe_makedir(self.test_dir)
#parse num of sessions from configuration
enroll_sessions = self.conf['enroll_sessions']
test_sessions = self.conf['test_sessions']
assert enroll_sessions+test_sessions <= 10,\
"The summation of all sessions must be less than or equal 10!!"
#iterate over all preprocessed waves
wav_filenames = os.listdir(self.data_dir)
for wav in tqdm(wav_filenames, desc="Copying enroll/test waves"):
_, sess, _, _ = wav.split(".")
inwav = os.path.join(self.data_dir, wav)
if int(sess) <= enroll_sessions:
outwav = os.path.join(self.enroll_dir, wav)
shutil.copyfile(inwav, outwav)
elif int(sess) <= enroll_sessions+test_sessions:
outwav = os.path.join(self.test_dir, wav)
shutil.copyfile(inwav, outwav)
def create_idMap(self, group):
"""
IdMap are used to store two lists of strings and to map between them.
Most of the time, IdMap are used to associate segments names (sessions)
stored in leftids; with the ID of their class (that could be the speaker
ID) stored in rightids.
Additionally, and in order to allow more flexibility, IdMap includes two
other vectors: 'start'and 'stop' which are float vectors used to store
boudaries of audio segments.
Args:
group (string): name of the group that we want to create idmap for
NOTE: Duplicated entries are allowed in each list.
"""
assert group in ["enroll", "test"],\
"Invalid group name!! Choose either 'enroll', 'test'"
# Make enrollment (IdMap) file list
group_dir = os.path.join(self.audio_dir, group)
group_files = sorted(os.listdir(group_dir))
# list of model IDs
group_models = [files.split('.')[0] for files in group_files]
# list of audio segments IDs
group_segments = [group+"/"+f for f in group_files]
# Generate IdMap
group_idmap = sidekit.IdMap()
group_idmap.leftids = np.asarray(group_models)
group_idmap.rightids = np.asarray(group_segments)
group_idmap.start = np.empty(group_idmap.rightids.shape, '|O')
group_idmap.stop = np.empty(group_idmap.rightids.shape, '|O')
if group_idmap.validate():
group_idmap.write(os.path.join(self.task_dir, group+'_idmap.h5'))
#generate tv_idmap and plda_idmap as well
if group == "enroll":
group_idmap.write(os.path.join(self.task_dir, 'tv_idmap.h5'))
group_idmap.write(os.path.join(self.task_dir, 'plda_idmap.h5'))
else:
raise RuntimeError('Problems with creating idMap file')
def create_test_trials(self):
"""
Ndx objects store trials index information, i.e., combination of
model and segment IDs that should be evaluated by the system which
will produce a score for those trials.
The trialmask is a m-by-n matrix of boolean where m is the number of
unique models and n is the number of unique segments. If trialmask(i,j)
is true then the score between model i and segment j will be computed.
"""
# Make list of test segments
test_data_dir = os.path.join(self.audio_dir, "test") #test data directory
test_files = sorted(os.listdir(test_data_dir))
test_files = ["test/"+f for f in test_files]
# Make lists for trial definition, and write to file
test_models = []
test_segments = []
test_labels = []
# Get enroll speakers
enrolled_speakers = set([])
for filename in os.listdir(os.path.join(self.audio_dir, "enroll")):
enrolled_speakers.add(filename.split(".")[0])
enrolled_speakers = sorted(enrolled_speakers)
for model in tqdm(enrolled_speakers, desc="Creating Test Cases"):
for segment in sorted(test_files):
test_model = segment.split(".")[0].split("/")[-1]
test_models.append(model)
test_segments.append(segment)
# Compare gender and speaker ID for each test file
if test_model == model:
test_labels.append('target')
else:
test_labels.append('nontarget')
with open(os.path.join(self.task_dir, "test_trials.txt"), "w") as fh:
for i in range(len(test_models)):
fh.write(test_models[i]+' '+test_segments[i]+' '+test_labels[i]+'\n')
def create_Ndx(self):
"""
Key are used to store information about which trial is a target trial
and which one is a non-target (or impostor) trial. tar(i,j) is true
if the test between model i and segment j is target. non(i,j) is true
if the test between model i and segment j is non-target.
"""
#Define Key and Ndx from text file
#SEE: https://projets-lium.univ-lemans.fr/sidekit/_modules/sidekit/bosaris/key.html
key = sidekit.Key.read_txt(os.path.join(self.task_dir, "test_trials.txt"))
ndx = key.to_ndx()
if ndx.validate():
ndx.write(os.path.join(self.task_dir, 'test_ndx.h5'))
else:
raise RuntimeError('Problems with creating idMap file')
def structure(self):
"""
This is the main method for this class, it calls all previous
methods... that's basically what it does :)
"""
self.preprocess_audio()
self.create_idMap("enroll")
self.create_idMap("test")
self.create_test_trials()
self.create_Ndx()
print("DONE!!")
if __name__ == "__main__":
conf_filename = "py3env/conf.yaml"
init = Initializer(conf_filename)
init.structure()