184 lines
6.5 KiB
Python
184 lines
6.5 KiB
Python
# Copyright (c) 2021 PaddlePaddle Authors. All Rights Reserved.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
|
|
from functools import partial
|
|
from typing import List
|
|
from pathlib import Path
|
|
import multiprocessing as mp
|
|
|
|
import numpy as np
|
|
from tqdm import tqdm
|
|
|
|
from audio_processor import SpeakerVerificationPreprocessor
|
|
|
|
|
|
def _process_utterance(path_pair, processor: SpeakerVerificationPreprocessor):
|
|
# Load and preprocess the waveform
|
|
input_path, output_path = path_pair
|
|
wav = processor.preprocess_wav(input_path)
|
|
if len(wav) == 0:
|
|
return
|
|
|
|
# Create the mel spectrogram, discard those that are too short
|
|
frames = processor.melspectrogram(wav)
|
|
if len(frames) < processor.partial_n_frames:
|
|
return
|
|
|
|
np.save(output_path, frames)
|
|
|
|
|
|
def _process_speaker(speaker_dir: Path,
|
|
processor: SpeakerVerificationPreprocessor,
|
|
datasets_root: Path,
|
|
output_dir: Path,
|
|
pattern: str,
|
|
skip_existing: bool=False):
|
|
# datastes root: a reference path to compute speaker_name
|
|
# we prepand dataset name to speaker_id becase we are mixing serveal
|
|
# multispeaker datasets together
|
|
speaker_name = "_".join(speaker_dir.relative_to(datasets_root).parts)
|
|
speaker_output_dir = output_dir / speaker_name
|
|
speaker_output_dir.mkdir(parents=True, exist_ok=True)
|
|
|
|
# load exsiting file set
|
|
sources_fpath = speaker_output_dir / "_sources.txt"
|
|
if sources_fpath.exists():
|
|
try:
|
|
with sources_fpath.open("rt") as sources_file:
|
|
existing_names = {line.split(",")[0] for line in sources_file}
|
|
except:
|
|
existing_names = {}
|
|
else:
|
|
existing_names = {}
|
|
|
|
sources_file = sources_fpath.open("at" if skip_existing else "wt")
|
|
for in_fpath in speaker_dir.rglob(pattern):
|
|
out_name = "_".join(
|
|
in_fpath.relative_to(speaker_dir).with_suffix(".npy").parts)
|
|
if skip_existing and out_name in existing_names:
|
|
continue
|
|
out_fpath = speaker_output_dir / out_name
|
|
_process_utterance((in_fpath, out_fpath), processor)
|
|
sources_file.write(f"{out_name},{in_fpath}\n")
|
|
|
|
sources_file.close()
|
|
|
|
|
|
def _process_dataset(processor: SpeakerVerificationPreprocessor,
|
|
datasets_root: Path,
|
|
speaker_dirs: List[Path],
|
|
dataset_name: str,
|
|
output_dir: Path,
|
|
pattern: str,
|
|
skip_existing: bool=False):
|
|
print(
|
|
f"{dataset_name}: Preprocessing data for {len(speaker_dirs)} speakers.")
|
|
|
|
_func = partial(
|
|
_process_speaker,
|
|
processor=processor,
|
|
datasets_root=datasets_root,
|
|
output_dir=output_dir,
|
|
pattern=pattern,
|
|
skip_existing=skip_existing)
|
|
|
|
with mp.Pool(16) as pool:
|
|
list(
|
|
tqdm(
|
|
pool.imap(_func, speaker_dirs),
|
|
dataset_name,
|
|
len(speaker_dirs),
|
|
unit="speakers"))
|
|
print(f"Done preprocessing {dataset_name}.")
|
|
|
|
|
|
def process_librispeech(processor,
|
|
datasets_root,
|
|
output_dir,
|
|
skip_existing=False):
|
|
dataset_name = "LibriSpeech/train-other-500"
|
|
dataset_root = datasets_root / dataset_name
|
|
speaker_dirs = list(dataset_root.glob("*"))
|
|
_process_dataset(processor, datasets_root, speaker_dirs, dataset_name,
|
|
output_dir, "*.flac", skip_existing)
|
|
|
|
|
|
def process_voxceleb1(processor,
|
|
datasets_root,
|
|
output_dir,
|
|
skip_existing=False):
|
|
dataset_name = "VoxCeleb1"
|
|
dataset_root = datasets_root / dataset_name
|
|
|
|
anglophone_nationalites = ["australia", "canada", "ireland", "uk", "usa"]
|
|
with dataset_root.joinpath("vox1_meta.csv").open("rt") as metafile:
|
|
metadata = [line.strip().split("\t") for line in metafile][1:]
|
|
|
|
# speaker id -> nationality
|
|
nationalities = {
|
|
line[0]: line[3]
|
|
for line in metadata if line[-1] == "dev"
|
|
}
|
|
keep_speaker_ids = [
|
|
speaker_id for speaker_id, nationality in nationalities.items()
|
|
if nationality.lower() in anglophone_nationalites
|
|
]
|
|
print(
|
|
"VoxCeleb1: using samples from {} (presumed anglophone) speakers out of {}."
|
|
.format(len(keep_speaker_ids), len(nationalities)))
|
|
|
|
speaker_dirs = list((dataset_root / "wav").glob("*"))
|
|
speaker_dirs = [
|
|
speaker_dir for speaker_dir in speaker_dirs
|
|
if speaker_dir.name in keep_speaker_ids
|
|
]
|
|
_process_dataset(processor, datasets_root, speaker_dirs, dataset_name,
|
|
output_dir, "*.wav", skip_existing)
|
|
|
|
|
|
def process_voxceleb2(processor,
|
|
datasets_root,
|
|
output_dir,
|
|
skip_existing=False):
|
|
dataset_name = "VoxCeleb2"
|
|
dataset_root = datasets_root / dataset_name
|
|
# There is no nationality in meta data for VoxCeleb2
|
|
speaker_dirs = list((dataset_root / "wav").glob("*"))
|
|
_process_dataset(processor, datasets_root, speaker_dirs, dataset_name,
|
|
output_dir, "*.wav", skip_existing)
|
|
|
|
|
|
def process_aidatatang_200zh(processor,
|
|
datasets_root,
|
|
output_dir,
|
|
skip_existing=False):
|
|
dataset_name = "aidatatang_200zh/train"
|
|
dataset_root = datasets_root / dataset_name
|
|
|
|
speaker_dirs = list((dataset_root).glob("*"))
|
|
_process_dataset(processor, datasets_root, speaker_dirs, dataset_name,
|
|
output_dir, "*.wav", skip_existing)
|
|
|
|
|
|
def process_magicdata(processor,
|
|
datasets_root,
|
|
output_dir,
|
|
skip_existing=False):
|
|
dataset_name = "magicdata/train"
|
|
dataset_root = datasets_root / dataset_name
|
|
|
|
speaker_dirs = list((dataset_root).glob("*"))
|
|
_process_dataset(processor, datasets_root, speaker_dirs, dataset_name,
|
|
output_dir, "*.wav", skip_existing)
|