-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathfakeflac.py
executable file
·93 lines (81 loc) · 2.24 KB
/
fakeflac.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
#!/usr/bin/python
# -*- coding: utf-8 -*-
# uncomment for debugging:
#import matplotlib.pyplot as plt
import numpy
from scipy.fftpack import rfft
from scipy.io.wavfile import read
from scipy.signal import hann
import sys
import warnings
def moving_average(a, w):
# calculate moving average
window = numpy.ones(int(w)) / float(w)
r = numpy.convolve(a, window, 'valid')
# len(a) = len(r) + w
a = numpy.empty((int(w / 2)))
a.fill(numpy.nan)
b = numpy.empty((int(w - len(a))))
b.fill(numpy.nan)
# add nan arrays to equal input and output length
return numpy.concatenate((a, r, b))
def find_cutoff(a, dx, diff, limit):
for i in range(1, int(a.shape[0] - dx)):
if a[-i] / a[-1] > limit:
break
if a[int(-i - dx)] - a[-i] > diff:
return a.shape[0] - i - dx
return a.shape[0]
# print usage if no argument given
if len(sys.argv[1:]) < 1:
print('usage %s audio_file.wav' % (sys.argv[0]))
sys.exit(1)
# read audio samples and ignore warnings, print errors
try:
with warnings.catch_warnings():
warnings.simplefilter('ignore')
input_data = read(sys.argv[1])
except IOError as e:
print(e[1])
sys.exit(e[0])
# process data
freq = input_data[0]
audio = input_data[1]
channel = 0
samples = len(audio[:, 0])
seconds = int(samples / freq)
seconds = min(seconds, 30)
spectrum = [0] * freq
# run over the seconds (max 30)
for t in range(0, seconds - 1):
# apply hanning window
window = hann(freq)
audio_second = audio[t * freq:(t + 1) * freq, channel] * window
# do fft to add second to frequency spectrum
spectrum += abs(rfft(audio_second))
# calculate average of the spectrum
spectrum /= seconds
# normalize frequency spectrum
spectrum = numpy.lib.scimath.log10(spectrum)
# smoothen frequency spectrum with window w
spectrum = moving_average(spectrum, freq / 100)
# find cutoff in frequency spectrum
cutoff = find_cutoff(spectrum, freq / 50, 1.25, 1.1)
# print percentage of frequency spectrum before cutoff
out = (int((cutoff * 100) / freq))
print(out)
if out == 100:
sys.exit(0)
else:
sys.exit(1)
# debugging only:
if 'plt' in globals():
# plot
plt.plot(spectrum)
# label the axes
plt.ylabel('Magnitude')
plt.xlabel('Frequency')
# set the title
plt.title('Spectrum')
plt.axis((0, 45000, 0, 10))
plt.show()