RtmpDiscovery-推流服务发现
一个可动态拓展的nginx rtmp流发现服务
仓库地址:RtmpDiscovery
具体功能
部署完毕后,可以动态注册nginx rtmp 服务器,并将服务器信息发布到指定的redis上
遇到的问题及解决方案
- 第一个遇到的问题是如何获取摄像头的视频流。
- 后面发现可以通过nginx搭建一个推流服务器。
- 第二个遇到的问题的如何处理摄像头的上线和下线。
- 刚开始是想让java后端和python算法后端都轮询nginx服务器,但是这样要维护两套轮询逻辑,而且对服务器压力较大。
- 于是我统一封装了轮询系统,即发现服务。发现服务来轮询摄像头状态,然后通过发布订阅来发布到redis服务器,之后两种后端只要订阅即可。
- 第三个遇到的问题是单个服务器带宽较小。
- 想法是通过部署多台推流服务器来分流。但是现有的技术好像比较难以实现对大带宽流量的分流。如nginx的负载均衡,实际上所有的流量都要经过负载均衡的服务器,适合对cpu压力大的场景。而其他的如LVS的TUN模式又要有硬件的支持。
- 之后想到转发rtmp协议,但是实现后发现只有打了补丁的rtmp客户端才能识别302转发,而一般的客户端都不支持。
- 最后我的实现方法是在每一台推流服务器上额外部署一个自己编写的服务,来将实际的服务器ip发布到redis,之后发现服务通过redis中的ip来发现服务器,之后在通过发现的服务器去发现在线的设备。而redis中的服务器地址都是真实的ip,然后设备再通过DNS来解析到不同ip来实现注册到不同的分流服务器上。
组件图
配置图
实现思路
概述
通过Rtmp推流发现服务客户端(rtmpDiscoveryClient)将nginx rtmp服务器的ip设置到redis中。之后Rtmp推流发现服务服务端(rtmpDiscoveryService)从redis中获取所有在线的服务器ip,通过这些ip去对应的nginx服务器轮询其rtmp流的信息。每次轮询都会将新获取到的信息与redis中已发布的信息对比,如果不相同就将服务器变更的信息发布到redis的channel中。
RtmpDiscoveryClient
client端首先要通过一个外网服务器来获取本机的公网ip,获取到ip之后,每过一段时间(默认是3s)就给redis服务器设置一个键值对,键值对也会在一段时间后过期(默认是9s)。如果服务端在键值对过期前没有重新设置(在这里就是连续三次没有设置存活),redis中的键值对就过期,service认为其下线。
RtmpDiscoveryService
service每次轮询首先从redis中获取所有在线服务器ip,之后一一访问约定的获取rtmp流信息的地址获取对应服务器上的流信息。最后整合,与redis之前存在的rtmp流信息对比,如果相同就略过,如果出现差异,就更新,并且发布订阅消息,通知所有订阅的客户端。
部署方式
client客户端要和nginx部署在同一设备上,然后获取ip的地址要改成对应service部署的地址
Nginx服务器配置
http{
server{
listen 80;
server_name 127.0.0.1;
#开启rtmp模块中自带的
location /stat {
rtmp_stat all;
}
location /live{
#Cors配置
add_header 'Access-Control-Allow-Origin' '*';
add_header 'Access-Control-Allow-Methods' 'GET, POST, OPTIONS, PUT, DELETE';
add_header 'Access-Control-Allow-Headers' 'Content-Type';
if ($request_method = 'OPTIONS') {
add_header 'Access-Control-Allow-Origin' '*';
add_header 'Access-Control-Max-Age' 1728000;
add_header 'Content-Type' 'text/plain charset=UTF-8';
add_header 'Content-Length' 0;
return 204;
}
alias /dev/shm/hls;
#自动生成index
autoindex on;
types {
application/dash+xml mpd;
application/vnd.apple.mpegurl m3u8;
video/mp2t ts;
}
add_header Cache-Control no-cache;
}
}
}
rtmp{
server{
listen 1953;
application rtmp{
live on;
}
application hls{
live on;
#启用hls
hls on;
#缓存地址
hls_path /dev/shm/hls/;
}
}
}
RtmpDiscoveryClient配置
启动
直接通过java -jar启动即可
自定义配置文件
#redis服务器中用来存储当前服务器ip的key前缀
rtmp.redis.ipKey=rtmp:ip:
#获取当前服务器公网ip的地址,可以使用Service服务器的部署地址的/get_client_ip方法
rtmp.service.url=http://...
#还要配置自己的redis
spring.redis.host=...
spring.redis.password=...
RtmpDiscoveryService配置
启动
直接通过java -jar启动即可
自定义配置文件
#获取的所有rtmp推流服务信息存储在redis中的key
rtmp.redis.key=rtmp:device
#在redis中发布服务器状态改变信息的channel
rtmp.redis.channel=deviceChannel
#获取所有在线的服务器ip的key前缀,与client的rtmp.redis.ipKey对应
rtmp.redis.ipKey=rtmp:ip:*
#还要配置自己的redis
spring.redis.host=...
spring.redis.password=...
示例
service服务器:http://121.43.147.139:8080/swagger
rtmp推流链接:rtmp://121.43.147.139:1953/*
或rtmp://106.15.74.153:1953/*
其中*
为自定义的名称,作为设备名
两台rtmp推流服务器是通过上面的部署方式部署的,设备接入就可以在网页看到状态更新
应用的项目
项目地址:https://github.com/FZUSESPR21W/STOP-Project