-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathpraat.py
61 lines (56 loc) · 2.03 KB
/
praat.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
from typing import Union
import numpy as np
import parselmouth
from config import Config
class PraatAugment:
"""Praat based augmentation.
"""
def __init__(self,
config: Config,
pitch_steps: float = 0.01,
pitch_floor: float = 75,
pitch_ceil: float = 600):
"""Initializer.
Args:
config: configurations.
pitch_steps: pitch measurement intervals.
pitch_floor: minimum pitch.
pitch_ceil: maximum pitch.
"""
self.config = config
self.pitch_steps = pitch_steps
self.pitch_floor = pitch_floor
self.pitch_ceil = pitch_ceil
def augment(self,
snd: Union[parselmouth.Sound, np.ndarray],
formant_shift: float = 1.,
pitch_shift: float = 1.,
pitch_range: float = 1.,
duration_factor: float = 1.) -> np.ndarray:
"""Augment the sound signal with praat.
"""
if not isinstance(snd, parselmouth.Sound):
snd = parselmouth.Sound(snd, sampling_frequency=self.config.model.sr)
pitch = parselmouth.praat.call(
snd, 'To Pitch', self.pitch_steps, self.pitch_floor, self.pitch_ceil)
ndpit = pitch.selected_array['frequency']
# if all unvoiced
nonzero = ndpit > 1e-5
if nonzero.sum() == 0:
return snd.values[0]
# if voiced
median, minp = np.median(ndpit[nonzero]).item(), ndpit[nonzero].min().item()
# scale
updated = median * pitch_shift
scaled = updated + (minp * pitch_shift - updated) * pitch_range
# for preventing infinite loop of `Change gender`
# ref:https://github.com/praat/praat/issues/1926
if scaled < 0.:
pitch_range = 1.
out, = parselmouth.praat.call(
(snd, pitch), 'Change gender',
formant_shift,
median * pitch_shift,
pitch_range,
duration_factor).values
return out