-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathQ3.py
126 lines (117 loc) · 3.41 KB
/
Q3.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
#!/usr/bin/python
# -*- coding:utf-8 -*-
from glob import glob
from collections import defaultdict
import librosa
import numpy as np
from scipy.stats import pearsonr
from mir_eval.key import weighted_score
# %%
import utils # self-defined utils.py file
DB = 'GTZAN'
if DB == 'GTZAN': # dataset with genre label classify at parent directory
FILES = glob(DB + '/wav/*/*.wav')
# print(FILES)
else:
FILES = glob(DB + '/wav/*.wav')
# print(FILES)
GENRE = [g.split('/')[2] for g in glob(DB + '/wav/*')]
print(GENRE)
n_fft = 100 # (ms)
hop_length = 25 # (ms)
# %% Q3
if DB == 'GTZAN':
label, pred = defaultdict(list), defaultdict(list)
else:
label, pred = list(), list()
chromagram = list()
gens = list()
for f in FILES:
f = f.replace('\\', '/')
print("file: ", f)
content = utils.read_keyfile(f, '*.lerch.txt')
if (int(content) < 0): continue # skip saving if key not found
if DB == 'GTZAN':
gen = f.split('/')[2]
label[gen].append(utils.LABEL[int(content)])
gens.append(gen)
else:
label.append(utils.LABEL[content])
sr, y = utils.read_wav(f)
cxx = librosa.feature.chroma_stft(y=y, sr=sr)
chromagram.append(cxx) # store into list for further use
chroma_vector = np.sum(cxx, axis=1)
key_ind = np.where(chroma_vector == np.amax(chroma_vector))
key_ind = int(key_ind[0])
# print('key index: ', key_ind)
chroma_vector = utils.rotate(chroma_vector.tolist(), 12 - key_ind)
# print('chroma_vector: ', chroma_vector)
MODE = {"major": [1, 0, 1, 0, 1, 1, 0, 1, 0, 1, 0, 1],
"minor": [1, 0, 1, 1, 0, 1, 0, 1, 1, 0, 1, 0]}
r_co_major = pearsonr(chroma_vector, MODE["major"])
r_co_minor = pearsonr(chroma_vector, MODE["minor"])
# print(r_co_major[0])
# print(r_co_minor[0])
mode = ''
if (r_co_major[0] > r_co_minor[0]):
mode = key_ind
else:
mode = key_ind + 12
mode = utils.lerch_to_str(mode)
# print('mode', mode)
if DB == 'GTZAN':
pred[gen].append(mode)
else:
pred.append('?') # you may ignore this when starting with GTZAN dataset
# print(pred[gen])
print("***** Q3 *****")
if DB == 'GTZAN':
label_list, pred_list = list(), list()
print("Genre \taccuracy")
for g in GENRE:
# TODO: Calculate the accuracy for each genre
# Hint: Use label[g] and pred[g]
sum = 0
for acc_len in range(len(label[g])):
score = weighted_score(label[g][acc_len], pred[g][acc_len])
sum += score
try:
acc = sum / len(label[g])
except ZeroDivisionError:
acc = 0
print("{:9s}\t{:8.2f}%".format(g, acc))
label_list += label[g]
pred_list += pred[g]
else:
label_list = label
pred_list = pred
# TODO: Calculate the accuracy for all file.
# Hint1: Use label_list and pred_list.
sum_all = 0
for acc_len in range(len(label_list)):
score_all = weighted_score(label_list[acc_len], pred_list[acc_len])
sum_all += score_all
try:
acc_all = sum_all / len(label_list)
except ZeroDivisionError:
acc_all = 0
##########
print("----------")
print("Overall accuracy:\t{:.2f}%".format(acc_all))
'''
GTZAN
***** Q3 *****
Genre accuracy
pop 0.57%
metal 0.35%
disco 0.49%
blues 0.20%
reggae 0.48%
classical -
rock 0.49%
hiphop 0.20%
country 0.55%
jazz 0.32%
----------
Overall accuracy: 0.41%
'''