此文章已经十分久远,请直接去看
【重写】树莓派驱动的b站点播台
最新的文章已经把代码更新为弹幕点歌了,并且在GitHub持续更新。
在这篇文章发布之前,我在it水家已经投过一稿了:http://www.ithome.com/html/win10/311694.htm
另外上一篇不带点歌功能,只有推流功能的教程:https://www.chenxublog.com/2017/06/02/raspi-live-24h-bilibili.html
如果是零基础,建议先看一遍上面那两篇文章,如果是使用代码,请参考本文。这几天代码已经更新多次。
我写的代码只能保证可用,毕竟我是业余的233333欢迎大神拿我的思路进行优化
github:https://github.com/chenxuuu/24h-raspberry-live-on-bilibili
php端运行的代码(最后更新于2017.6.4晚):
<head> <meta charset="utf-8"> </head> <body> <form action="" method="post"> <p>一个简陋的点歌台23333</p> <p>搜索结果来自网易云</p> <p>输入歌曲名搜索歌曲:<input type="text" name="song" /></p> <p><input type="submit" name="sub" value="搜索" /></p> </form> <form action="" method="post"> <p>或者直接输入id点歌(推荐!)<br/>实例(红色部分为id):<br/>http://music.163.com/song/<font color="red">26489014</font>/?userid=261620056<br/>http://music.163.com/#/song?id=<font color="red">32477053</font></p> <p>id:<input type="text" name="id" /></p> <p><input type="submit" name="sub" value="查看" /></p> </form> by 晨旭/chenxublog.com | running on Raspberry Pi 2 Model B<br/>直播间地址:http://live.bilibili.com/16703<br/><br/><br/> 当前状态:<div id='t'></div> <?php require_once 'NeteaseMusicAPI_mini.php'; function get_url_id($id) { $api = new NeteaseMusicAPI(); $result = $api->url($id); $result = str_replace("[","",$result); $result = str_replace("]","",$result); $data=json_decode($result, true); return $data['data']['url']; } function urlcheck($url) { $counttemp=0; for($counttemp=1;$counttemp<31;$counttemp++) { if (file_exists($counttemp.'.txt') && $url==file_get_contents($counttemp.'.txt')) { return false; } } return true; } function netease_http($url) { $refer = "http://music.163.com/"; $header[] = "Cookie: " . "appver=1.5.0.75771;"; $ch = curl_init(); curl_setopt($ch, CURLOPT_URL, $url); curl_setopt($ch, CURLOPT_HTTPHEADER, $header); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); curl_setopt($ch, CURLOPT_BINARYTRANSFER, true); curl_setopt($ch, CURLOPT_REFERER, $refer); $cexecute = curl_exec($ch); curl_close($ch); if ($cexecute) { $result = json_decode($cexecute, true); return $result; } else { return false; } } function getStatus($url){ if($headers = @get_headers($url)){ $status = $headers[0]; $statusno= false; if(preg_match_all('%HTTP/1\.1 ([\d]{3})%i',$status,$matches)){ $statusno = $matches[1][0]; } return $statusno; } return false; } if(!empty($_POST['song'])){ $song=$_POST['song']; echo '搜索关键词:'.$song.'<br/>=========================================================<br/>'; $url = "http://s.music.163.com/search/get/?type=1&s=".$song; $response = netease_http($url); $counttemp=0; for($counttemp=0;$counttemp<10;$counttemp++) { if(getStatus($response['result']['songs'][$counttemp]['audio'])=='200') { echo $response['result']['songs'][$counttemp]['name'].'<img src="'.$response['result']['songs'][$counttemp]['album']['picUrl'].'" height="100"/><a href="?down='.$response['result']['songs'][$counttemp]['audio'].'">就选这首了</a><br/>试听:<audio src="'.$response['result']['songs'][$counttemp]['audio'].'" controls="controls"></audio><br/>=========================================================<br/>'; } else { echo '第'.($counttemp+1).'首歌曲获取失败(歌曲失效/版权问题/网络抽风)<br/>=========================================================<br/>'; } } } elseif(!empty($_POST['id'])){ echo '试听:<audio src="'.get_url_id($_POST['id']).'" controls="controls"></audio><br/><a href="?down='.get_url_id($_POST['id']).'">就选这首了</a>'; } elseif(!empty($_GET['down']) && empty($_GET['write']) && strpos($_GET['down'],"music.126.net")!=false && getStatus($_GET['down'])=='200') { echo '当前可供选择的空闲的序列(注意,直播是从第1首到第30首按顺序播放的。歌曲长度最大限制六分钟,超过将丢弃。):<br/>'; $counttemp=0; for($counttemp=1;$counttemp<31;$counttemp++) { if($counttemp%5==0) { echo '<br/>'; } if (!file_exists($counttemp.'.txt')) { echo '<a href="?write='.$counttemp.'&down='.$_GET['down'].'">换掉第'.$counttemp.'首歌</a>(<font color="green">可以换歌</font>)|'; } else { echo '第'.$counttemp.'首歌(<font color="red">排队渲染中</font>)|'; } } } elseif(!empty($_GET['write']) && !empty($_GET['down']) && urlcheck($_GET['down']) && getStatus($_GET['down'])=='200') { if(!file_exists($_GET['write'].'.txt')) { //file_put_contents('songs/'.$_GET['write'].'.txt', $_GET['down']); $myfile = fopen($_GET['write'].'.txt', "w") or die("Unable to open file!"); fwrite($myfile, $_GET['down']); fclose($myfile); echo '第'.$_GET['write'].'首歌曲的渲染请求提交成功!<a href="index.php">返回首页</a>'; } else { echo '啊哦!这个序号已经有人选中了!正在渲染!'; } } else { echo '树莓当前负载状态(注意,直播是从第1首到第30首按顺序播放的):<br/>'; $counttemp=0; for($counttemp=1;$counttemp<31;$counttemp++) { if($counttemp%5==0) { echo '<br/>'; } if (!file_exists($counttemp.'.txt')) { echo '第'.$counttemp.'首歌(<font color="green">可以换掉</font>)|'; } else { echo '第'.$counttemp.'首歌(<font color="red">排队渲染中</font>)|'; } } } ?> <script type="text/javascript" src="now.js"></script> </body>
同目录下需要新建一个NeteaseMusicAPI_mini.php,文件取自:
https://github.com/metowolf/NeteaseCloudMusicApi/blob/master/weapi/NeteaseMusicAPI_mini.php
php文件我全部放到了/usr/share/nginx/www/songs目录下。
渲染处理部分:
ff.py(路径/home/pi/songs/)
# -*- coding:utf-8 -*- import os import urllib import eyed3 import time for i in range(1, 30+1): if(os.path.exists(str(i)+'.mp3')): os.remove(str(i)+'.mp3') os.remove('/usr/share/nginx/www/songs/'+str(i)+'.txt') if(os.path.exists('/usr/share/nginx/www/songs/'+str(i)+'.txt')): f = open('/usr/share/nginx/www/songs/'+str(i)+'.txt') content = f.read() fileout = file('/usr/share/nginx/www/songs/now.js','w') fileout.write('t.innerHTML=("正在下载'+str(i)+'.mp3")') fileout.close() urllib.urlretrieve(content, str(i)+'.mp3') print('download success') xx=eyed3.load(str(i)+'.mp3') seconds=xx.info.time_secs if(seconds<600): fileout = file('/usr/share/nginx/www/songs/now.js','w') fileout.write('t.innerHTML=("正在生成'+str(i)+'.mp4的一图流视频 第一步/共两步")') fileout.close() os.system('ffmpeg -loop 1 -r 1 -t '+str(seconds)+' -f image2 -i '+str(i)+'.png -vcodec libx264 -pix_fmt yuv420p -crf 24 -y SinglePictureVideo.mp4') fileout = file('/usr/share/nginx/www/songs/now.js','w') fileout.write('t.innerHTML=("正在将'+str(i)+'.flv的视频与音频合为一体 第二步/共两步")') fileout.close() os.system('ffmpeg -i SinglePictureVideo.mp4 -i '+str(i)+'.mp3 -c:v copy -c:a aac -y '+str(i)+'.flv') os.remove(str(i)+'.mp3') os.remove('SinglePictureVideo.mp4') os.remove('/usr/share/nginx/www/songs/'+str(i)+'.txt') fileout = file('/usr/share/nginx/www/songs/now.js','w') fileout.write('t.innerHTML=("成功渲染'+str(i)+'.flv!60秒后会开始渲染下一个视频")') fileout.close() else: os.remove(str(i)+'.mp3') os.remove('/usr/share/nginx/www/songs/'+str(i)+'.txt') time.sleep(60) time.sleep(10)
ff.sh(路径/home/pi/songs/)
#!/bin/bash while true do python ff.py done
路径/home/pi/songs/下还有1-30.png文件用于作为视频内容
启动ffmpeg部分:
playlist.txt请自己写吧,规则见https://trac.ffmpeg.org/wiki/Concatenate
live.sh(与playlist.txt同级)
#!/bin/bash while true do ffmpeg -re -f concat -safe 0 -i playlist.txt -vcodec copy -acodec aac -b:a 192k -f flv "你的直播地址和码" done
理论上到这里就结束了,但是网络老断,那我只能暴力检测和强制重启推流来解决了
断网自动重推部分:
net.py
#coding:utf8 '''python3 code author's email: [email protected] 通过统计ifconfig命令的输出,计算当前网速 ''' import logging logging.basicConfig(level=logging.INFO, format='%(message)s', #filename='speed', #filemod='w' ) import os, sys, time import re def get_total_tx_bytes(interface, isCN): grep = '发送字节' if isCN else '"TX bytes"' r = os.popen('ifconfig '+interface+' | grep '+grep).read() total_bytes = re.sub('(.+:)| \(.+','',r) return int(total_bytes) def get_total_rx_bytes(interface, isCN): grep = '接收字节' if isCN else '"RX bytes"' r = os.popen('ifconfig '+interface+' | grep '+grep).read() total_bytes = re.sub(' \(.+','',r) total_bytes = re.sub('.+?:','',total_bytes) return int(total_bytes) if __name__=='__main__': interface = sys.argv[1] get_total_bytes = get_total_tx_bytes if sys.argv[2]=='tx' else get_total_rx_bytes isCN = True if sys.argv[3]=='cn' else False freq = int(sys.argv[4]) low_count = 0 for i in range(1, 10 + 1): last = get_total_bytes(interface, isCN) time.sleep(freq) increase = get_total_bytes(interface, isCN) - last logging.info(str(increase/freq/1000)) speed_now = increase/freq/1000 if(speed_now < 5): low_count = low_count +1 if(low_count > 5): os.system('killall ffmpeg') time.sleep(1) os.system('killall ffmpeg') time.sleep(1) os.system('killall ffmpeg') logging.info('666') else: logging.info('ok!')
net.sh
#!/bin/bash while true do python net.py eth0 tx false 1 done
这样所有我用的脚本都贴在这里了,启动方式就是扔几个screen就行了
screen sh live.sh #按ctrl+a,ctrl+d screen sh net.sh #按ctrl+a,ctrl+d cd songs/ screen sh ff.sh #按ctrl+a,ctrl+d
完毕~
技术不高,如有错误,请指出,谢谢o( ̄▽ ̄)ブ
好厉害,不过我没有树莓派,用VPS搭建可以吗?
理论上应该也是可行的,你可以试试
呃,ffmpeg编译不成功,想问问博主的树莓派装的是什么系统呢?
试试yum或者apt-get看看有没有?
树莓系统是基于debian的raspbian,ubuntu之类的都是基于debian
我换了很多个系统,Debian和Ubuntu都用过了,就是在安装解码器时执行
./configure –host=arm-unknown-linux-gnueabi –enable-static –disable-opencl
会提示No working C compiler found.,我谷歌了也找不到有效的解决方法,请问你有没有遇到这种情况?
https://cn.bing.com/search?q=No%20working%20C%20compiler%20found.
搜技术相关的东西,请用必应。
yum 安装下GCC就好了 因为系统没有gcc 无法编译
我用的centos 安装上了ffmpeg 可以的 但在配置nginx的时候一直出各种各样的问题
树莓派实验室转载了,特来感谢一下,多谢分享。
http://shumeipai.nxez.com/2017/06/11/use-raspberry-pi-to-build-bilibili-vod.html
哇,第一次看见符合cc协议的转载?
大神,我的CPU消耗为什么一直要90%以上?FFmpeg还能再优化吗?
如果有条件。。。开超频吧。。。
记得加上散热风扇和散热铜片
好像有个h264_omx硬解码,可当初编译FFmpeg时候没有开启。用重编FFmpeg心好累啊。
h264_omx有帮助的吧。看到有人树莓派直播CPU占用只用了5%。
参考博主的搭建了一个能点播MV的:https://aoaoao.me/live/index.php 直播地址:http://live.bilibili.com/3368800
要是能解决切换视频时卡死的问题就好了,这样就能随便怎么玩了?
怎么始终是在排队渲染中就没换过状态呢!
ff.py那个渲染处理脚本定时运行了吗
厉害了
代码高亮体验极差,无法复制,只能复制单行,因为每行代码是个单独的标签,平板、电脑均为此状
解决方案1:点一下到别的页面,再点回来
解决方案2:没发现可以双击复制全部代码的吗
双击没提示,而且也很奇怪,不顺手,像其他网站右上角一键复制比较好。点到别的页面我无力吐槽。关了js我才复制的。真心不推荐一行一个标签
反正双击过后会出现一个单独的文本框,默认全选
代码高亮是编辑器自带的,我没装过代码高亮插件