这是从基础闹铃功能上进行的升级, 关于基础闹铃请看这里 —- 用 Node.js + Raspberry Pi 打造一个不一样的闹钟。
做出最基本的功能(到时间就 “哔-哔-哔–” 的叫)后,突然有一天,觉得每天早上和晚上播放一段天气预报是件很好玩的事儿。
音频准备
播放天气预报最主要的是天气预报的语音,我能想到的来源:
一种是使用天气查询网站的接口,获取到数据,然后:
- 使用一些语音库合成语音 — 效果会很差
- 使用在线的一些语音合成服务合成语音 — 效果会好点
- 使用一些第三方的天气预报的语音包,进行语音合成 — 效果会更好点
不过这种,都是机器合成的语音,不灵活,很僵硬。一定找真人播放的才行,于是我想到了:
- 电视台的天气预报的录音
- FM 电台的的天气预报的录音
- 互联网电台是不是也有天气预报?
顺着思路就去喜马拉雅找了一下,发现真有一个叫“中国天气 show” 的节目,每天在更新北京的天气预报,而且是小哥哥小姐姐们的真人播报,声音也好听。这简直是为我专门准备的呀。
接下来就是要拿到这个节目里的音频。
获取音频
首先我发现了一个获取播放列表的接口,可以搞到音频地址:
接口: https://www.ximalaya.com/revision/play/album?albumId=22689810&pageNum=1&sort=-1&pageSize=10
接口数据如下:
{
"ret": 200,
"msg": "声音播放数据",
"data": {
"albumId": 22689810,
"sort": 1,
"pageNum": 1,
"pageSize": 5,
"tracksAudioPlay": [
{
"trackName": "20190427早间 北京-降雨降温大风 晓丹",
"albumName": "中国天气show|北京天气",
"albumUrl": "/toutiao/22689810/",
"src": "http://aod.tx.xmcdn.com/group58/M07/DE/13/wKgLc1zDj93Dvc4jAAdzvJEZxrM644.m4a",
"updateTime": "8小时前",
// ...
},
// ...
]
}
}
其中,data.tracksAudioPlay[0].src
便是我们需要的音频。
http://audio.xmcdn.com/group58/M07/DE/13/wKgLc1zDj93Dvc4jAAdzvJEZxrM644.m4a
本来这个接口是好使的,不过也就用了三天,就不好使了,请求到的数据成了下面一句话:
[SIGN] no sign or wrong sign
这时喜马拉雅开始在请求头里加一些签名验证了,一个叫 xm-sign
的字段,值是类似这么一坨东西:
xm-sign: ebe5ffe394af8693bf35383ebb3c034d(6)1558191544160(81)1558191547336
这个值每次都不一样,而且同样的一个 xm-sign
值,过一会也会失效。
我大概研究了一下这一坨东西的生成规则,感觉以我的智商估计搞不定。这时候我想到了终极神器 — pupeteer。
pupeteer
先说结果,使用 pupeteer 在树莓派上失败了。过程如下:
我在 ubuntu 上写的程序是能跑的,但是拿到树莓派上总是超时。
这只能说喜马拉雅的 Web 端做的真的不咋地,在 chrome 中打开,到页面资源加载完毕,要等很久。在 pupeteer 打开页面时,则需要更久,而在树莓派中的 pupeteer 打开,需要更更久。设置超时两分钟还不行,我就放弃了。
可能与我的树莓派比较旧(树莓派 2B)有关吧。无论如何,得找其他方式,直到我发现了这个接口:
http://m.ximalaya.com/m-revision/page/album/queryAlbumPage/22689810
这个接口很大,但是里面可以拿到音频的地址。
不过,我有种预感,因为很多无良的爬虫利用这些接口在疯狂地爬取喜马拉雅的音频,当上面提到的那个接口使用的成本变高之后,这个接口会被越来越的人发现并利用,到时候这个接口的请求也会需要签名验证了。
播放音频
拿到音频地址之后,就是使用树莓派播放了。
使用到的工具 mplayer, 一款 Linux 上强大的播放器。
mplayer
- 安装
sudo apt-get install mplayer
- 使用
mplayr xxx.mp3
- 在 Nodejs 中使用
const exec = require('child_process').exec;
exec('mplayer xxx.mp3');
闹铃加入天气预报功能
利用上面介绍的基本原理,在闹铃项目新增 speaker 模块,其中导出一个 weather 方法给上层调用。
改一下原来的逻辑,在第一次需要闹铃的时候播放天气预报,如果请求出错,才会响铃。
speaker.wether().catch(() => {
alerm(5, '', hour < 12);
});
最后
这里只是着重介绍了探索过程和基本原理,而几乎没有涉及源码。如果对源码感兴趣可以看我的GitHub。