forked from raspberry-pi-org/voiceAssistant
-
Notifications
You must be signed in to change notification settings - Fork 1
/
voiceAssistant.py
226 lines (201 loc) · 7.35 KB
/
voiceAssistant.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
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
import snowboydecoder
import signal
import wave
import sys
import json
import requests
import time
import os
import base64
from pyaudio import PyAudio, paInt16
import webbrowser
from fetchToken import fetch_token
import sys
# 打印所有 python 解释器可以搜索到的所有路径
sys.path.append('../')
print(sys.path)
# 导入自定义包
from plugins.baidu.baiduTopNews import baiduTopNews # 播报百度今日热点事件
from plugins.weibo.weiboHotTopic import weiboHotTopic # 播报微博热门话题
from plugins.countdown.countdown import countdown # 播报距离 2021 年考研初试的倒计时
from plugins.douban.getMoveData import moviceData # 播报豆瓣近一周电影口碑榜
from plugins.maoyan.getMoveDataMY import moviceDataMY # 播报猫眼今日票房排行榜及正在热映的电影
from plugins.weather.weatherInfo import weatherInfo # 播报今日或明日天气预报
from plugins.system.sysinfo import sysinfo # 播报系统信息数据, 如 ip 地址及 cpu 利用率等..
interrupted = False # snowboy监听唤醒结束标志
endSnow = False # 程序结束标志
framerate = 16000 # 采样率
num_samples = 2000 # 采样点
channels = 1 # 声道
sampwidth = 2 # 采样宽度2bytes
FILEPATH = './audio/audio.wav' # 录制完成存放音频路径
TTS_URL = 'http://tsn.baidu.com/text2audio' # 文字转语音接口
music_exit = './audio/exit.wav' # 唤醒系统退出语音
music_open = './audio/open.wav' # 唤醒系统打开语音
# os.close(sys.stderr.fileno())
def signal_handler(signal, frame):
"""
监听键盘结束
"""
global interrupted
interrupted = True
def interrupt_callback():
"""
监听唤醒
"""
global interrupted
return interrupted
def detected():
"""
唤醒成功
"""
print('唤醒成功')
play('./audio/open.wav')
global interrupted
interrupted = True
detector.terminate()
def play(filename):
"""
播放音频
"""
wf = wave.open(filename, 'rb') # 打开audio.wav
p = PyAudio() # 实例化 pyaudio
# 打开流
stream = p.open(format=p.get_format_from_width(wf.getsampwidth()),
channels=wf.getnchannels(),
rate=wf.getframerate(),
output=True)
data = wf.readframes(1024)
while data != b'':
data = wf.readframes(1024)
stream.write(data)
# 释放IO
stream.stop_stream()
stream.close()
p.terminate()
def save_wave_file(filepath, data):
"""
存储文件
"""
wf = wave.open(filepath, 'wb')
wf.setnchannels(channels)
wf.setsampwidth(sampwidth)
wf.setframerate(framerate)
wf.writeframes(b''.join(data))
wf.close()
def my_record():
"""
录音
"""
pa = PyAudio()
stream = pa.open(format=paInt16, channels=channels,
rate=framerate, input=True, frames_per_buffer=num_samples)
my_buf = []
# count = 0
t = time.time()
print('开始录音...')
while time.time() < t + 4: # 秒
string_audio_data = stream.read(num_samples)
my_buf.append(string_audio_data)
print('录音结束!')
save_wave_file(FILEPATH, my_buf)
stream.close()
def speech2text(speech_data, token, dev_pid=1537):
"""
音频转文字
"""
FORMAT = 'wav'
RATE = '16000'
CHANNEL = 1
CUID = 'baidu_workshop'
SPEECH = base64.b64encode(speech_data).decode('utf-8')
data = {
'format': FORMAT,
'rate': RATE,
'channel': CHANNEL,
'cuid': CUID,
'len': len(speech_data),
'speech': SPEECH,
'token': token,
'dev_pid': dev_pid
}
url = 'https://vop.baidu.com/pro_api'
headers = {'Content-Type': 'application/json'}
print('正在识别...')
r = requests.post(url, json=data, headers=headers)
Result = r.json()
if 'result' in Result:
return Result['result'][0]
else:
return Result
def get_audio(file):
"""
获取音频文件
"""
with open(file, 'rb') as f:
data = f.read()
return data
def identifyComplete(text):
"""
识别成功
"""
print('识别内容成功,内容为:' + text)
maps = {
'打开百度': ['打开百度。', '打开百度', '打开百度,', 'baidu'],
'播放音乐': ['播放音乐。', '播放音乐', '播放音乐,', 'yinyue'],
'百度热点事件': ['百度热点事件。', '百度热点事件,', '百度热点事件', '百度新闻,', '百度新闻。', '百度新闻'],
'微博热门话题': ['微博热门话题。', '微博热门话题,', '微博热门话题', '微博话题,', '微博话题。', '微博话题'],
'考研倒计时': ['考研倒计时。', '考研倒计时,', '考研倒计时', '考研,', '考研。', '考研'],
'豆瓣电影排行榜': ['豆瓣电影排行榜。', '豆瓣电影排行榜,', '豆瓣电影排行榜', '豆瓣电影,', '豆瓣电影。', '豆瓣电影'],
'猫眼电影排行榜': ['猫眼电影排行榜。', '猫眼电影排行榜,', '猫眼电影排行榜'],
'猫眼热映电影': ['猫眼热映电影,', '猫眼热映电影。', '猫眼热映电影'],
'今日天气预报': ['今日天气预报,', '今日天气预报。', '今日天气预报', '今日天气,', '今日天气。', '今日天气'],
'明日天气预报': ['明日天气预报,', '明日天气预报。', '明日天气预报', '明日天气,', '明日天气。', '明日天气'],
'系统信息': ['系统信息,', '系统信息。', '系统信息', 'IP地址,', 'IP地址。', 'IP地址'],
}
if (text == '再见。' or text == '拜拜。'):
play(music_exit) # 关闭系统播放反馈语音
exit()
if text in maps['打开百度']:
webbrowser.open_new_tab('https://www.baidu.com')
# play('./audio/openbaidu.wav') # 识别到播放反馈语音
print('baidu opened bro!')
if text in maps['播放音乐']:
play('./audio/LoveYourself.wav')
print('music played bro!')
if text in maps['百度热点事件']:
baiduTopNews.run()
if text in maps['微博热门话题']:
weiboHotTopic.run()
if text in maps['考研倒计时']:
countdown.run()
if text in maps['豆瓣电影排行榜']:
moviceData.run()
if text in maps['猫眼电影排行榜']:
moviceDataMY.run_today_movice_ranking()
if text in maps['猫眼热映电影']:
moviceDataMY.run_movice_ranking()
if text in maps['今日天气预报']:
weatherInfo.run_today_weather()
if text in maps['明日天气预报']:
weatherInfo.run_tomorrow_weather()
if text in maps['系统信息']:
sysinfo.run()
else:
play('./audio/none.wav') # 未匹配口令播放反馈语音
print('操作完成')
if __name__ == "__main__":
print('enter main')
while endSnow == False:
interrupted = False
detector = snowboydecoder.HotwordDetector('resources/snowboy.umdl', sensitivity=0.5)
print('等待唤醒')
detector.start(detected_callback=detected,
interrupt_check=interrupt_callback,
sleep_time=0.03)
my_record()
TOKEN = fetch_token()
speech = get_audio(FILEPATH)
result = speech2text(speech, TOKEN, int(80001))
if type(result) == str:
identifyComplete(result)