需求
前段时间公司有个需求,将无人机流接入到第三方可视化平台中。无人机流使用的是华大疆提供的rtmp流,而第三方可视化平台不支持rmtp流,仅支持flv协议流。因此经过多方查找,找到了wvp和zlmediakit的开源方案实现对rtmp协议流转为flv协议流。下面是详细的安装流程。
zlmediakit和wvp的安装踩过居多坑,从Ubuntu到windows再到centos。前前后后折腾了小一个月,所有平台的流程都走通了,因此写这篇博客作为centos的踩坑记录。
Step1 docker安装
由于公司的生产环境以centos7.9为主,在官方文档中又不推荐使用centos7.x的版本,因此最终的解决方案选择了在服务器中使用docker容器进行服务部署。(centos7.x的坑我踩完了,哭死,千万别用,不然一堆坑)
docker的安装参考了这篇博客,这个博主也有对docker的使用教程,大家可以去学习一下。大家参考他的博客就能安装docker了,很简单。然后就是docker镜像问题,要找到合适的docker镜像源,后续才能拉取镜像。
step2 redis安装
在安装完全docker后,使用如下命令从官方镜像中拉取redis
docker pull redis:7.2
这个版本号可以随意,但是不建议太新的,怕出bug。
在拉取完镜像后,可以使用docker images
查看是否安装成功镜像。
最后再根据如下命令启动容器:
docker run -it --name redis7.2 -p 6379:6379 -d redis --restart=always --network host
这里面最重要的是--network
,我直接使用了宿主机的网络,没有使用docker的虚拟网桥。如果使用虚拟网桥,那么后续就会出现注册国标信息的地址是虚拟网桥地址,不是你服务器地址。(这地方我也是踩过坑的,哭死)
step3 Mysql安装
这里就不用过多赘述了,按照如下命令运行就行了。
docker pull mysql:8.0
启动mysql
docker run -d -p 3306:3306 --privileged=true --network host -v /home/mysql/log:/var/log/mysql -v /home/mysql/data:/var/lib/mysql -v /home/mysql/conf:/etc/mysql/conf.d -e MYSQL_ROOT_PASSWORD=你的Mysql密码 --name mysql8.0 mysql:8.0
注意,一定要改mysql密码,就是上面MYSQL_ROOT_PASSWORD的字段!!!!网络也是直接使用宿主机网络。
在这里设置好了之后就能用Navicat直接连接这个mysql了,但是由于公司网络端口的问题,我就一直没连上,我还以为是我没设置好的问题,折腾了一两天。
然后进入mysql运行的容器,
docker exec -it <你的mysql容器id> /bin/bash
执行如下命令创建wvp数据库:
CREATE DATABASE wvp CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci;
查看数据库:
show databases;
切换为wvp库:
use wvp;
将数据库转为wvp数据库后使用如下名利初始化源文件自带的表:
source /home/mysql/inital-mysql-2.7.3.sql;
至此,你已经完成了比较重要的内容,mysql和redis,并且在后台运行。接下来就是自己写dockerfile编译自己需要的环境了。
step4 dockerfile构建
在参考了几位大佬的博客并得到大佬的指点后,将官方的dockerfile进行修改,得到如下的文件形式。
FROM ubuntu:20.04 as build
RUN apt-get update && apt-get install -y libcurl4 && apt-get clean
RUN apt-get install make
ARG gitUrl="https://gitee.com/pan648540858"
ARG zlmGitUrl="https://gitee.com/xia-chu/ZLMediaKit"
RUN export DEBIAN_FRONTEND=noninteractive &&\
apt-get update && \
apt-get install -y --no-install-recommends openjdk-11-jre git maven nodejs npm build-essential \
cmake ca-certificates openssl ffmpeg &&\
mkdir -p /opt/wvp/config /opt/wvp/heapdump /opt/wvp/config /opt/assist/config /opt/assist/heapdump /opt/media/www/record
RUN cd /home && \
git clone "${gitUrl}/maven.git" && \
cp maven/settings.xml /usr/share/maven/conf/
RUN cd /home && \
git clone "${gitUrl}/wvp-GB28181-pro.git"
RUN cd /home/wvp-GB28181-pro/web_src && \
npm install && \
npm run build
RUN cd /home/wvp-GB28181-pro && \
mvn clean package -Dmaven.test.skip=true && \
cp /home/wvp-GB28181-pro/target/*.jar /opt/wvp/ && \
cp /home/wvp-GB28181-pro/src/main/resources/application-docker.yml /opt/wvp/config/application.yml
RUN cd /home && \
git clone "${gitUrl}/wvp-pro-assist.git"
RUN cd /home/wvp-pro-assist && \
git reset --hard 58f1a79136a55a7cd1593c95b56ddadcc2225b61 && \
mvn clean package -Dmaven.test.skip=true && \
cp /home/wvp-pro-assist/target/*.jar /opt/assist/ && \
cp /home/wvp-pro-assist/src/main/resources/application-dev.yml /opt/assist/config/application.yml
RUN cd /home && \
git clone --depth=1 "${zlmGitUrl}"
RUN cd /home/ZLMediaKit && \
git submodule update --init --recursive && \
mkdir -p build release/linux/Release/ &&\
cd build && \
cmake -DCMAKE_BUILD_TYPE=Release .. && \
make -j4 && \
rm -rf ../release/linux/Release/config.ini && \
cp -r ../release/linux/Release/* /opt/media
#RUN cd /opt/wvp && \
# echo '#!/bin/bash' > run.sh && \
# echo 'echo ${WVP_IP}' >> run.sh && \
# echo 'echo ${WVP_CONFIG}' >> run.sh && \
# echo 'cd /opt/assist' >> run.sh && \
# echo 'nohup java ${ASSIST_JVM_CONFIG} -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=/opt/assist/heapdump/ -jar *.jar --spring.config.location=/opt/assist/config/application.yml --userSettings.record=/opt/media/www/record/ --media.record-assist-port=18081 ${ASSIST_CONFIG} &' >> run.sh && \
# echo 'nohup /opt/media/MediaServer -d -m 3 &' >> run.sh && \
# echo 'cd /opt/wvp' >> run.sh && \
# echo 'java ${WVP_JVM_CONFIG} -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=/opt/wvp/heapdump/ -jar *.jar --spring.config.location=/opt/wvp/config/application.yml --media.record-assist-port=18081 ${WVP_CONFIG}' >> run.sh && \
# chmod +x run.sh
FROM ubuntu:20.04
EXPOSE 18080/tcp
EXPOSE 5060/tcp
EXPOSE 5060/udp
EXPOSE 6379/tcp
EXPOSE 18081/tcp
EXPOSE 80/tcp
EXPOSE 1935/tcp
EXPOSE 554/tcp
EXPOSE 554/udp
EXPOSE 30000-30500/tcp
EXPOSE 30000-30500/udp
ENV LC_ALL zh_CN.UTF-8
RUN export DEBIAN_FRONTEND=noninteractive &&\
apt-get update && \
apt-get install -y --no-install-recommends openjdk-11-jre ca-certificates ffmpeg language-pack-zh-hans && \
apt-get autoremove -y && \
apt-get clean -y && \
rm -rf /var/lib/apt/lists/*dic
COPY --from=build /opt /opt
WORKDIR /opt/wvp
#CMD ["sh", "run.sh"]
在我后续的使用过程中发现这个dockerfile好像有点问题,不能在运行docker容器的时候直接使用里面的内容(笑哭)。不过影响不大,能构建成功。
把这个文件写入到宿主机中,并命名为official.Dockerfile
在终端执行如下命令就能安心等待镜像构建成功了。
docker build -t official/wvp_gb28181 -f official.Dockerfile
经历过前面的几步骤就有下面红框三个镜像(请自动忽略其他的,我之前测试的,没删除)
step5 wvp + zlmediakit运行
在构建完上述镜像后通过如下命令启动:
docker run -d -it --name='wvp_pro' --network host --env WVP_IP=宿主机IP -p 18080:18080 -p 30000-30500:30000-30500/udp -p 30000-30500:30000-30500/tcp -p 80:80 -p 5060:5060 -p 5060:5060/udp official/wvp_gb28181:1.0
WVP_IP记得换成自己宿主机的IP。
启动后所有镜像如下图所示:
在这之前将zlmediakit的config.ini文件和wvp的application.yml文件进行修改。可以参考一下我的config.ini文件和wvp-application.yml文件
config.ini
; auto-generated by mINI class {
[api]
apiDebug=1
defaultSnap=./www/logo.png
downloadRoot=./www
secret=8aQ3lSPkSiUNhG0Sf94P5Fm3ZJep1d8S
snapRoot=./www/snap/
[cluster]
origin_url=
retry_count=3
timeout_sec=15
[ffmpeg]
bin=/usr/bin/ffmpeg
cmd=%s -re -i %s -c:a aac -strict -2 -ar 44100 -ab 48k -c:v libx264 -f flv %s
log=./ffmpeg/ffmpeg.log
restart_sec=0
snap=%s -rtsp_transport tcp -i %s -y -f mjpeg -frames:v 1 %s
[general]
broadcast_player_count_changed=0
check_nvidia_dev=1
enableVhost=0
enable_ffmpeg_log=0
flowThreshold=1024
maxStreamWaitMS=15000
mediaServerId=WestBayIntelligence
mergeWriteMS=0
resetWhenRePlay=1
streamNoneReaderDelayMS=20000
unready_frame_cache=100
wait_add_track_ms=3000
wait_track_ready_ms=10000
[hls]
broadcastRecordTs=0
deleteDelaySec=10
fastRegister=0
fileBufSize=65536
segDelay=0
segDur=2
segKeep=0
segNum=3
segRetain=5
[hook]
alive_interval=10.0
enable=1
on_flow_report=
on_http_access=
on_play=http://你的宿主机ip:18080/index/hook/on_play
on_publish=http://你的宿主机ip:18080/index/hook/on_publish
on_record_mp4=http://你的宿主机ip:18080/index/hook/on_record_mp4
on_record_ts=
on_rtp_server_timeout=http://你的宿主机ip:18080/index/hook/on_rtp_server_timeout
on_rtsp_auth=
on_rtsp_realm=
on_send_rtp_stopped=http://你的宿主机ip:18080/index/hook/on_send_rtp_stopped
on_server_exited=
on_server_keepalive=http://你的宿主机ip:18080/index/hook/on_server_keepalive
on_server_started=http://你的宿主机ip:18080/index/hook/on_server_started
on_shell_login=
on_stream_changed=http://你的宿主机ip:18080/index/hook/on_stream_changed
on_stream_none_reader=http://你的宿主机ip:18080/index/hook/on_stream_none_reader
on_stream_not_found=http://你的宿主机ip:18080/index/hook/on_stream_not_found
retry=1
retry_delay=3.0
stream_changed_schemas=rtsp/rtmp/fmp4/ts/hls/hls.fmp4
timeoutSec=30
[http]
allow_cross_domains=1
allow_ip_range=::1,127.0.0.1,172.16.0.0-172.31.255.255,192.168.0.0-192.168.255.255,10.0.0.0-10.255.255.255
charSet=utf-8
dirMenu=1
forbidCacheSuffix=
forwarded_ip_header=
keepAliveSecond=30
maxReqSize=40960
notFound=<html><head><title>404 Not Found</title></head><body bgcolor="white"><center><h1>您访问的资源不存在!</h1></center><hr><center>ZLMediaKit(git hash:c6a8118/2024-07-19T17:10:22+08:00,branch:master,build time:2024-07-22T10:43:52)</center></body></html>
port=9080
rootPath=./www
sendBufSize=65536
sslport=443
virtualPath=
[multicast]
addrMax=239.255.255.255
addrMin=239.0.0.0
udpTTL=64
[protocol]
add_mute_audio=1
auto_close=0
continue_push_ms=3000
enable_audio=1
enable_fmp4=1
enable_hls=1
enable_hls_fmp4=0
enable_mp4=0
enable_rtmp=1
enable_rtsp=1
enable_ts=1
fmp4_demand=0
hls_demand=0
hls_save_path=./www
modify_stamp=2
mp4_as_player=0
mp4_max_second=3600
mp4_save_path=./www
paced_sender_ms=0
rtmp_demand=0
rtsp_demand=0
ts_demand=0
[record]
appName=record
enableFmp4=0
fastStart=0
fileBufSize=65536
fileRepeat=0
sampleMS=500
[rtc]
externIP=
maxNackMS=5000
max_bitrate=0
min_bitrate=0
nackIntervalRatio=1.0
nackMaxCount=15
nackMaxMS=3000
nackMaxSize=2048
nackRtpSize=8
port=8000
preferredCodecA=PCMA,PCMU,opus,mpeg4-generic
preferredCodecV=H264,H265,AV1,VP9,VP8
rembBitRate=0
rtpCacheCheckInterval=100
start_bitrate=0
tcpPort=8000
timeoutSec=15
[rtmp]
directProxy=1
enhanced=0
handshakeSecond=15
keepAliveSecond=15
port=1935
sslport=0
[rtp]
audioMtuSize=600
h264_stap_a=1
lowLatency=0
rtpMaxSize=10
videoMtuSize=1400
[rtp_proxy]
dumpDir=
gop_cache=1
h264_pt=98
h265_pt=99
opus_pt=100
port=10000
port_range=50000-50300
ps_pt=96
rtp_g711_dur_ms=100
timeoutSec=15
udp_recv_socket_buffer=4194304
[rtsp]
authBasic=0
directProxy=1
handshakeSecond=15
keepAliveSecond=15
lowLatency=0
port=554
rtpTransportType=-1
sslport=0
[shell]
maxReqSize=1024
port=0
[srt]
latencyMul=4
pktBufSize=8192
port=9000
timeoutSec=5
; } ---
wvp-application.yml
spring:
# 设置接口超时时间
mvc:
async:
request-timeout: 20000
thymeleaf:
cache: false
# [可选]上传文件大小限制
servlet:
multipart:
max-file-size: 10MB
max-request-size: 100MB
cache:
type: redis
# REDIS数据库配置
redis:
# [必须修改] Redis服务器IP, REDIS安装在本机的,使用127.0.0.1
host: 你的宿主机ip
# [必须修改] 端口号
port: 6379
# [可选] 数据库 DB
database: 7
# [可选] 访问密码,若你的redis服务器没有设置密码,就不需要用密码去连接
password:
# [可选] 超时时间
timeout: 10000
# mysql数据源
datasource:
dynamic:
primary: master
datasource:
master:
type: com.zaxxer.hikari.HikariDataSource
driver-class-name: com.mysql.cj.jdbc.Driver
# 数据库连接要修改
url: jdbc:mysql://你的宿主机ip:3306/wvp?useUnicode=true&characterEncoding=UTF8&rewriteBatchedStatements=true&serverTimezone=PRC&useSSL=false&allowMultiQueries=true
username: root
password: mysql数据库ip
hikari:
connection-timeout: 20000 # 是客户端等待连接池连接的最大毫秒数
initialSize: 50 # 连接池初始化连接数
maximum-pool-size: 200 # 连接池最大连接数
minimum-idle: 10 # 连接池最小空闲连接数
idle-timeout: 300000 # 允许连接在连接池中空闲的最长时间(以毫秒为单位)
max-lifetime: 1200000 # 是池中连接关闭后的最长生命周期(以毫秒为单位)
#[可选] WVP监听的HTTP端口, 网页和接口调用都是这个端口
server:
port: 18080
# [可选] HTTPS配置, 默认不开启
ssl:
# [可选] 是否开启HTTPS访问
enabled: false
# [可选] 证书文件路径,放置在resource/目录下即可,修改xxx为文件名
key-store: classpath:test.monitor.89iot.cn.jks
# [可选] 证书密码
key-store-password: gpf64qmw
# [可选] 证书类型, 默认为jks,根据实际修改
key-store-type: JKS
# 作为28181服务器的配置
sip:
# [可选] 28181服务监听的端口
ip: 10.21.1.243
port: 5060
# 根据国标6.1.2中规定,domain宜采用ID统一编码的前十位编码。国标附录D中定义前8位为中心编码(由省级、市级、区级、基层编号组成,参照GB/T 2260-2007)
# 后两位为行业编码,定义参照附录D.3
# 3701020049标识山东济南历下区 信息行业接入
# [可选]
domain: 4101050000
# [可选]
id: 41010500002000000001
# [可选] 默认设备认证密码,后续扩展使用设备单独密码, 移除密码将不进行校验
password: bajiuwulian1006
# 是否存储alarm信息
alarm: true
#zlm 默认服务器配置
media:
id: WestBayIntelligence #自定义的ip,影响不大
# [必须修改] zlm服务器的内网IP
ip: 你宿主机ip
# [可选] 有公网IP就配置公网IP, 不可用域名
wan_ip:
# [必须修改] zlm服务器的http.port
http-port: 9080
# [可选] zlm服务器访问WVP所使用的IP, 默认使用127.0.0.1,zlm和wvp没有部署在同一台服务器时必须配置
hook-ip: 你宿主机ip
# [必选选] zlm服务器的hook.admin_params=secret
secret: 8aQ3lSPkSiUNhG0Sf94P5Fm3ZJep1d8S #要和config.ini的[api]的secret一致
# 启用多端口模式, 多端口模式使用端口区分每路流,兼容性更好。 单端口使用流的ssrc区分, 点播超时建议使用多端口测试
rtp:
# [可选] 是否启用多端口模式, 开启后会在portRange范围内选择端口用于媒体流传输
enable: true
# [可选] 在此范围内选择端口用于媒体流传输, 必须提前在zlm上配置该属性,不然自动配置此属性可能不成功
port-range: 50000,50300 # 端口范围
# [可选] 国标级联在此范围内选择端口发送媒体流,
send-port-range: 50000,50300 # 端口范围
# [根据业务需求配置]
user-settings:
# 点播/录像回放 等待超时时间,单位:毫秒
play-timeout: 180000
# [可选] 自动点播, 使用固定流地址进行播放时,如果未点播则自动进行点播, 需要rtp.enable=true
auto-apply-play: true
# 推流直播是否录制
record-push-live: true
# 国标是否录制
record-sip: true
# 国标点播 按需拉流, true:有人观看拉流,无人观看释放, false:拉起后不自动释放
stream-on-demand: true
修改自己的config.ini和wvp-application.yml文件。
进入到上面启动的wvp_pro的容器中
docker exec -it 容器ID /bin/bash
然后使用docker cp命令拷贝config.ini
和wvp-application.yml
进容器指定位置,其实就是对应的MediaServer文件位置和wvp-pro-2.7.3-09050220.jar对应的位置。
然后逐步运行这两个命令,就能启动zlmediakit和wvp了。
这个是在MediaServer的文件夹中
./MediaServer -c ./config.ini
这个是在jar包的文件夹中
java -jar wvp-pro-2.7.3-09050220.jar --spring.config.location=wvp-application.yml
当然,如果你想结束后台又两种方法,如果终端还有,就直接Ctrl + C就行,如果终端重新启动了,就要按照如下方式启动后台。
终止java进程
ps -ef | grep java
kill -9 PID
查看zlmediakit进程
ps aux | grep zlmediakit
终止MediaServer进程
killall -2 MediaServer
在经历九九八一难之后,就可以启动对应的页面了,宿主机IP:18080
。
在启动终端后发现wvp和zlmediakit对应的日志的时间不对,那么就要进行如下操作了。
docker cp /usr/share/zoneinfo/Asia/Shanghai 容器ID:etc/localtime
echo "Asia/shanghai" > /etc/timezone;
这个部署,自己也是借鉴了很多人的经验,最后才能成功的,摸索了巨久。
参考链接
docker安装redis Docker安装redis docker安装Redis 详细教程
assist + wvp-pro + zlm使用docker部署(全网最详细版)
【GB28181】wvp-GB28181-pro++ZLMediaKit编译安装国标流媒体服务器教程(Windows平台)