[변경사항]
- 피처를 최대한 많이 생성하여 데이터를 가공함 -> drop out 효과 극대화 및 과적합 방지, 정확도 향상
- 이전 기능 별 분리되었던 모델(ver 2.*) 을 통합함
- 각 데이터 별 193 피처 추출
- row 통일 안함 (3~4sec)
- 원핫인코딩 안함
- 라벨을 1차원 배열로 변경 -> 카테고리 별 int값 출력하게 함
[기존과 동일]
- 기능 1,2에 대해 사용 데이터는 뉴욕대학교 MARL의 URBANSOUND8K DATASET 일부와 일상 생활에서 녹음한 녹음 파일(.wav)를 활용 (2,622개, 1.96G)
- 기능 3에 대해 환승역 알림음을 트리거로 적용 (894개, 0.71G)
import numpy as np
import pandas as pd
#wav 파일들의 피처 생성
#librosa 사용
#사용 특성은 mfcc, chroma_stft, melspectorgram, spectral_contrast, tonnetz로 총193
#딥러닝 모델만 사용할 예정 -> 피처 축소 생략
import glob
import librosa
# 오디오 불러오기 + 피쳐 생성
# 피쳐 193개
# row 통일 안시킴
def extract_feature(file_name):
X, sample_rate = librosa.load(file_name)
stft = np.abs(librosa.stft(X))
mfccs = np.mean(librosa.feature.mfcc(y=X, sr=sample_rate, n_mfcc=40).T,axis=0)
chroma = np.mean(librosa.feature.chroma_stft(S=stft, sr=sample_rate).T,axis=0)
mel = np.mean(librosa.feature.melspectrogram(X, sr=sample_rate).T,axis=0)
contrast = np.mean(librosa.feature.spectral_contrast(S=stft, sr=sample_rate).T,axis=0)
tonnetz = np.mean(librosa.feature.tonnetz(y=librosa.effects.harmonic(X), sr=sample_rate).T,axis=0)
return mfccs,chroma,mel,contrast,tonnetz
#데이터 가공
#행렬로 변환
def parse_audio_files(filenames):
rows = len(filenames)
# feature는 각 파일 별 row(window) * 피처 의 2차원 행렬
# labels은 파일 별 카테고리 int 값
features, labels = np.zeros((rows,193)), np.zeros((rows, 1))
i = 0
for fn in filenames:
try:
mfccs, chroma, mel, contrast,tonnetz = extract_feature(fn)
ext_features = np.hstack([mfccs,chroma,mel,contrast,tonnetz])
y_col = int(fn.split('-')[0])
except:
print("error : "+fn)
else:
features[i] = ext_features
labels[i] = y_col
print(y_col)
i += 1
return features, labels
audio_files = []
#0 : 사이렌
#1 : 자동차가 다가오는 소리(엔진소리)
#2 : 자동차 경적소리
#4 : 환승역 안내음
audio_files.extend(glob.glob('*.wav'))
print(len(audio_files))
files = audio_files
X, y= parse_audio_files(files)
#?.npz
np.savez('0', X=X, y=y)
[변경사항]
- 자동차 경적 소리와 엔진 소리를 구분
- 다양한 차량 소리에 대해 데이터 셋 더욱 세분화
[기존과 동일]
- 파일 열람 쉽도록 npz로 가공
- 0.npz : 사이렌 등 알림음 (기능1)
- 1.npz : 자동차 경적소리 (기능2)
- 2.npz : 자동차 엔진소리 (기능2)
- 3.npz : 지하철 트리거 (기능3)
import numpy as np
import pandas as pd
import glob
import librosa
X = np.empty((0, 193))
y = np.empty((0, 1))
npz_files = glob.glob('?.npz')
for fn in npz_files:
print(fn)
model_1 = np.load(fn)
X = np.append(X, model_1['X'], axis=0)
y = np.append(y, model_1['y'], axis=0)
print(X.shape, y.shape)
np.savez('model_1', X=X, y=y)
[변경사항]
[기존과 동일]
- model_1.npz 불러오기
import glob
import librosa
import numpy as np
import pandas as pd
import matplotlib.pyplot as plt
# 알림, 차량 엔진, 차량 경적, 지하철 트리거 순
sound_data = np.load('model_1.npz')
X_train = sound_data['X']
y_train = sound_data['y']
X_train.shape, y_train.shape
X_train.shape, y_train.shape
[변경사항]
- lstm 사용
- 이전 모델의 경우 파라미터 조정에 초점을 두었으나 ver 3.5에서는 layer 구성에 초점을 두어 진행
[기존과 동일]
import tensorflow as tf
from tensorflow import keras
from tensorflow.keras.utils import to_categorical
import numpy as np
import matplotlib.pyplot as plt
import os
from keras import models
from keras import layers
from keras.layers import *
from keras import optimizers
from keras.layers import LSTM
from keras.models import Sequential
from keras.layers import Dense
import keras.backend as K
from keras.callbacks import EarlyStopping
K.clear_session()
model = Sequential() # Sequeatial Model
model.add(LSTM(20, input_shape=(193, 1))) # (timestep, feature)
model.add(Dense(1)) # output = 1
model.compile(loss='mean_squared_error', optimizer='adam')
model.summary()
#X_train = X_train.values
X_train = X_train.reshape(X_train.shape[0], 193, 1)
early_stop = EarlyStopping(monitor='loss', patience=1, verbose=1)
model.fit(X_train, y_train, epochs=100,
batch_size=30, verbose=1, callbacks=[early_stop])
[변경사항]
- pkl, json, pb, tflite로 저장
[기존과 동일]
# 모델 pkl로 저장하기
import joblib
joblib.dump(model, 'model/pkl/model_1.pkl')
# 모델 json으로 저장하기
model_1 = model.to_json()
# model = model_from_json(json_string)
# 모델 h5로 저장하기
from keras.models import load_model
model.save('model/h5/model_1')
model.save('model/h5/model_1.h5')
# 모델 pb로 저장하기
model = keras.models.load_model('model/h5/model_1', compile=False)
model.save('model/pb/',save_format=tf)
#모델 tflite 로 저장하기
saved_model_dir='model/pb/'
converter = tf.lite.TFLiteConverter.from_saved_model(saved_model_dir)
converter.target_spec.supported_ops=[tf.lite.OpsSet.TFLITE_BUILTINS,
tf.lite.OpsSet.SELECT_TF_OPS]
tfilte_mode=converter.convert()
open('model/tflite/model_1.tflite','wb').write(ftlite_model)
[변경사항]
- 4초로 제한
[기존과 동일]
- 블루투스 이어폰의 외부 마이크 사용 (채널 조정)
import librosa
import scipy.signal as signal
import numpy as np
import pandas as pd
import joblib
#환경 확인
import pyaudio
import wave
FORMAT = pyaudio.paInt16
CHANNELS = 1
RATE = 16000
CHUNK = 1024
RECORD_SECONDS = 4
WAVE_OUTPUT_FILENAME = "test_file.wav"
audio = pyaudio.PyAudio()
# start Recording
stream = audio.open(format=pyaudio.paInt16,
channels=CHANNELS,
rate=RATE,
input=True,
input_device_index=1,
frames_per_buffer=CHUNK)
print ("recording...")
frames = []
for i in range(0, int(RATE / CHUNK * RECORD_SECONDS)):
data = stream.read(CHUNK)
frames.append(data)
print ("finished recording")
stream.stop_stream()
stream.close()
audio.terminate()
waveFile = wave.open(WAVE_OUTPUT_FILENAME, 'wb')
waveFile.setnchannels(CHANNELS)
waveFile.setsampwidth(audio.get_sample_size(FORMAT))
waveFile.setframerate(RATE)
waveFile.writeframes(b''.join(frames))
waveFile.close()
[변경사항]
- 모듈화 할 것
[기존과 동일]
import numpy as np
import pandas as pd
import glob
def extract_feature(file_name):
X, sample_rate = librosa.load(file_name)
stft = np.abs(librosa.stft(X))
mfccs = np.mean(librosa.feature.mfcc(y=X, sr=sample_rate, n_mfcc=40).T,axis=0)
chroma = np.mean(librosa.feature.chroma_stft(S=stft, sr=sample_rate).T,axis=0)
mel = np.mean(librosa.feature.melspectrogram(X, sr=sample_rate).T,axis=0)
contrast = np.mean(librosa.feature.spectral_contrast(S=stft, sr=sample_rate).T,axis=0)
tonnetz = np.mean(librosa.feature.tonnetz(y=librosa.effects.harmonic(X), sr=sample_rate).T,axis=0)
return mfccs,chroma,mel,contrast,tonnetz
def parse_audio_files(filenames):
rows = len(filenames)
# feature는 각 파일 별 row(window) * 피처 의 2차원 행렬
# labels은 파일 별 카테고리 int 값
features = np.zeros((rows,193))
i = 0
for fn in filenames:
try:
mfccs, chroma, mel, contrast,tonnetz = extract_feature(fn)
ext_features = np.hstack([mfccs,chroma,mel,contrast,tonnetz])
except:
print("error : "+fn)
else:
features[i] = ext_features
print("성공")
i += 1
return features
audio_files = []
audio_files.extend(glob.glob(WAVE_OUTPUT_FILENAME))
print(len(audio_files))
files = audio_files
X_test = parse_audio_files(files)
[변경사항]
[기존과 동일]
- 0 : 사이렌, 민방위 등 알림음
- 1,2 : 차량 경적, 엔진소리
- 3 : 환승 트리거
X_test = X_test.reshape(X_test.shape[0], 193, 1)
model = joblib.load('model_1.pkl')
pred = model.predict_proba(X_test)
ans = float(pred)
print(pred)
def print_ans(ans):
ans = round(ans)
#기능 1
if(ans==0):
print("사이렌")
# 기능 2
if(ans==1):
print("차량 엔진 소리_접근중")
if(ans==2):
print("차량 경적 소리_위험")
# 기능 3 -> 모델 2 반환
if(ans==3):
print("지하철 트리거")
print_ans(ans)