2018-11-02 15:13:51 +00:00
|
|
|
'''
|
|
|
|
Extract spectrograms and save them to file for training
|
|
|
|
'''
|
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
import time
|
|
|
|
import glob
|
|
|
|
import argparse
|
|
|
|
import librosa
|
|
|
|
import importlib
|
|
|
|
import numpy as np
|
|
|
|
import tqdm
|
|
|
|
from utils.generic_utils import load_config, copy_config_file
|
|
|
|
from utils.audio import AudioProcessor
|
|
|
|
|
|
|
|
from multiprocessing import Pool
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
parser.add_argument('--data_path', type=str, help='Data folder.')
|
|
|
|
parser.add_argument('--cache_path', type=str, help='Cache folder, place to output all the intermediate spectrogram files.')
|
|
|
|
# parser.add_argument('--keep_cache', type=bool, help='If True, it keeps the cache folder.')
|
|
|
|
# parser.add_argument('--hdf5_path', type=str, help='hdf5 folder.')
|
|
|
|
parser.add_argument(
|
|
|
|
'--config', type=str, help='conf.json file for run settings.')
|
|
|
|
parser.add_argument(
|
|
|
|
"--num_proc", type=int, default=8, help="number of processes.")
|
|
|
|
parser.add_argument(
|
|
|
|
"--trim_silence",
|
|
|
|
type=bool,
|
|
|
|
default=False,
|
|
|
|
help="trim silence in the voice clip.")
|
|
|
|
parser.add_argument("--only_mel", type=bool, default=False, help="If True, only melsceptrogram is extracted.")
|
|
|
|
parser.add_argument("--dataset", type=str, help="Target dataset to be processed.")
|
|
|
|
parser.add_argument("--val_split", type=int, default=0, help="Number of instances for validation.")
|
|
|
|
parser.add_argument("--meta_file", type=str, help="Meta data file to be used for the dataset.")
|
|
|
|
parser.add_argument("--process_audio", type=bool, default=False, help="Preprocess audio files.")
|
|
|
|
args = parser.parse_args()
|
|
|
|
|
|
|
|
DATA_PATH = args.data_path
|
|
|
|
CACHE_PATH = args.cache_path
|
|
|
|
CONFIG = load_config(args.config)
|
|
|
|
|
|
|
|
# load the right preprocessor
|
|
|
|
preprocessor = importlib.import_module('datasets.preprocess')
|
|
|
|
preprocessor = getattr(preprocessor, args.dataset.lower())
|
|
|
|
items = preprocessor(args.data_path, args.meta_file)
|
|
|
|
|
|
|
|
print(" > Input path: ", DATA_PATH)
|
|
|
|
print(" > Cache path: ", CACHE_PATH)
|
|
|
|
|
|
|
|
# audio = importlib.import_module('utils.' + c.audio_processor)
|
|
|
|
# AudioProcessor = getattr(audio, 'AudioProcessor')
|
|
|
|
ap = AudioProcessor(**CONFIG.audio)
|
|
|
|
|
|
|
|
def trim_silence(self, wav):
|
|
|
|
""" Trim silent parts with a threshold and 0.1 sec margin """
|
|
|
|
margin = int(ap.sample_rate * 0.1)
|
|
|
|
wav = wav[margin:-margin]
|
|
|
|
return librosa.effects.trim(
|
|
|
|
wav, top_db=40, frame_length=1024, hop_length=256)[0]
|
|
|
|
|
|
|
|
def extract_mel(item):
|
|
|
|
""" Compute spectrograms, length information """
|
|
|
|
text = item[0]
|
|
|
|
file_path = item[1]
|
|
|
|
x = ap.load_wav(file_path, ap.sample_rate)
|
|
|
|
if args.trim_silence:
|
|
|
|
x = trim_silence(x)
|
|
|
|
file_name = os.path.basename(file_path).replace(".wav", "")
|
|
|
|
mel_file = file_name + "_mel"
|
|
|
|
mel_path = os.path.join(CACHE_PATH, 'mel', mel_file)
|
|
|
|
mel = ap.melspectrogram(x.astype('float32')).astype('float32')
|
|
|
|
np.save(mel_path, mel, allow_pickle=False)
|
|
|
|
mel_len = mel.shape[1]
|
|
|
|
wav_len = x.shape[0]
|
|
|
|
output = [file_path, mel_path+".npy", str(wav_len), str(mel_len), text]
|
|
|
|
if not args.only_mel:
|
|
|
|
linear_file = file_name + "_linear"
|
|
|
|
linear_path = os.path.join(CACHE_PATH, 'linear', linear_file)
|
|
|
|
linear = ap.spectrogram(x.astype('float32')).astype('float32')
|
|
|
|
linear_len = linear.shape[1]
|
|
|
|
np.save(linear_path, linear, allow_pickle=False)
|
|
|
|
output.insert(2, linear_path+".npy")
|
2018-12-18 13:50:44 +00:00
|
|
|
assert mel_len == linear_len
|
2018-11-02 15:13:51 +00:00
|
|
|
if args.process_audio:
|
|
|
|
audio_file = file_name + "_audio"
|
|
|
|
audio_path = os.path.join(CACHE_PATH, 'audio', audio_file)
|
|
|
|
np.save(audio_path, x, allow_pickle=False)
|
|
|
|
del output[0]
|
|
|
|
output.insert(0, audio_path+".npy")
|
|
|
|
return output
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
print(" > Number of files: %i" % (len(items)))
|
|
|
|
if not os.path.exists(CACHE_PATH):
|
|
|
|
os.makedirs(os.path.join(CACHE_PATH, 'mel'))
|
|
|
|
if not args.only_mel:
|
|
|
|
os.makedirs(os.path.join(CACHE_PATH, 'linear'))
|
|
|
|
if args.process_audio:
|
|
|
|
os.makedirs(os.path.join(CACHE_PATH, 'audio'))
|
|
|
|
print(" > A new folder created at {}".format(CACHE_PATH))
|
|
|
|
|
|
|
|
# Extract features
|
|
|
|
r = []
|
|
|
|
if args.num_proc > 1:
|
|
|
|
print(" > Using {} processes.".format(args.num_proc))
|
|
|
|
with Pool(args.num_proc) as p:
|
|
|
|
r = list(
|
|
|
|
tqdm.tqdm(
|
|
|
|
p.imap(extract_mel, items),
|
|
|
|
total=len(items)))
|
|
|
|
# r = list(p.imap(extract_mel, file_names))
|
|
|
|
else:
|
|
|
|
print(" > Using single process run.")
|
|
|
|
for item in items:
|
|
|
|
print(" > ", item[1])
|
|
|
|
r.append(extract_mel(item))
|
|
|
|
|
|
|
|
# Save meta data
|
|
|
|
if args.cache_path is not None:
|
|
|
|
file_path = os.path.join(CACHE_PATH, "tts_metadata_val.csv")
|
|
|
|
file = open(file_path, "w")
|
|
|
|
for line in r[:args.val_split]:
|
|
|
|
line = "| ".join(line)
|
|
|
|
file.write(line + '\n')
|
|
|
|
file.close()
|
|
|
|
|
|
|
|
file_path = os.path.join(CACHE_PATH, "tts_metadata.csv")
|
|
|
|
file = open(file_path, "w")
|
|
|
|
for line in r[args.val_split:]:
|
|
|
|
line = "| ".join(line)
|
|
|
|
file.write(line + '\n')
|
|
|
|
file.close()
|
|
|
|
|
|
|
|
# copy the used config file to output path for sanity
|
|
|
|
copy_config_file(args.config, CACHE_PATH)
|