ZLMediaKit/tests/test_server.cpp

203 lines
7.3 KiB
C++
Raw Normal View History

2017-10-09 22:11:01 +08:00
/*
2017-09-27 16:20:30 +08:00
* MIT License
*
* Copyright (c) 2016 xiongziliang <771730766@qq.com>
*
* This file is part of ZLMediaKit(https://github.com/xiongziliang/ZLMediaKit).
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
2017-04-05 09:39:16 +08:00
2017-08-09 18:39:30 +08:00
#include <map>
2017-04-05 09:39:16 +08:00
#include <signal.h>
#include <iostream>
2017-08-09 18:39:30 +08:00
#include "Common/config.h"
2017-04-05 09:39:16 +08:00
#include "Rtsp/UDPServer.h"
#include "Rtsp/RtspSession.h"
#include "Rtmp/RtmpSession.h"
#include "Http/HttpSession.h"
2017-08-10 16:16:42 +08:00
#include "Shell/ShellSession.h"
2017-12-10 01:34:43 +08:00
#include "Util/MD5.h"
2017-05-03 00:12:51 +08:00
#ifdef ENABLE_OPENSSL
2017-04-19 17:47:07 +08:00
#include "Util/SSLBox.h"
2017-05-03 00:12:51 +08:00
#include "Http/HttpsSession.h"
#endif//ENABLE_OPENSSL
2017-08-09 18:39:30 +08:00
#include "Util/File.h"
2017-04-25 11:35:41 +08:00
#include "Util/logger.h"
#include "Util/onceToken.h"
#include "Network/TcpServer.h"
#include "Poller/EventPoller.h"
#include "Thread/WorkThreadPool.h"
2017-05-02 23:43:10 +08:00
#include "Device/PlayerProxy.h"
2017-08-09 18:39:30 +08:00
2017-04-05 09:39:16 +08:00
using namespace std;
2017-09-30 13:00:12 +08:00
using namespace ZL::DEV;
2017-04-05 09:39:16 +08:00
using namespace ZL::Util;
using namespace ZL::Http;
using namespace ZL::Rtsp;
using namespace ZL::Rtmp;
2017-08-10 16:16:42 +08:00
using namespace ZL::Shell;
2017-04-05 09:39:16 +08:00
using namespace ZL::Thread;
using namespace ZL::Network;
2017-12-10 01:34:43 +08:00
#define REALM "realm_zlmedaikit"
static onceToken s_token([](){
NoticeCenter::Instance().addListener(nullptr,Config::Broadcast::kBroadcastOnGetRtspRealm,[](BroadcastOnGetRtspRealmArgs){
if(string("1") == stream ){
// live/1需要认证
EventPoller::Instance().async([invoker](){
//该流需要认证并且设置realm
invoker(REALM);
});
}else{
//我们异步执行invoker。
//有时我们要查询redis或数据库来判断该流是否需要认证通过invoker的方式可以做到完全异步
EventPoller::Instance().async([invoker](){
//该流我们不需要认证
invoker("");
});
}
});
NoticeCenter::Instance().addListener(nullptr,Config::Broadcast::kBroadcastOnRtspAuth,[](BroadcastOnRtspAuthArgs){
InfoL << "用户:" << user_name << (must_no_encrypt ? " Base64" : " MD5" )<< " 方式登录";
string user = user_name;
//假设我们异步读取数据库
EventPoller::Instance().async([must_no_encrypt,invoker,user](){
if(user == "test0"){
//假设数据库保存的是明文
invoker(false,"pwd0");
return;
}
if(user == "test1"){
//假设数据库保存的是密文
auto encrypted_pwd = MD5(user + ":" + REALM + ":" + "pwd1").hexdigest();
invoker(true,encrypted_pwd);
return;
}
if(user == "test2" && must_no_encrypt){
//假设登录的是test2,并且以base64方式登录此时我们提供加密密码那么会导致认证失败
//可以通过这个方式屏蔽base64这种不安全的加密方式
invoker(true,"pwd2");
return;
}
//其他用户密码跟用户名一致
invoker(false,user);
});
});
}, nullptr);
2017-04-05 09:39:16 +08:00
int main(int argc,char *argv[]){
2017-09-30 13:00:12 +08:00
//设置退出信号处理函数
signal(SIGINT, [](int){EventPoller::Instance().shutdown();});
//设置日志
2017-04-05 09:39:16 +08:00
Logger::Instance().add(std::make_shared<ConsoleChannel>("stdout", LTrace));
2017-09-30 13:00:12 +08:00
Logger::Instance().setWriter(std::make_shared<AsyncLogWriter>());
//加载配置文件,如果配置文件不存在就创建一个
2017-09-27 14:52:34 +08:00
Config::loadIniConfig();
2017-09-30 13:00:12 +08:00
//这里是拉流地址支持rtmp/rtsp协议负载必须是H264+AAC
//如果是其他不识别的音视频将会被忽略(譬如说h264+adpcm转发后会去除音频)
2017-05-02 23:43:10 +08:00
auto urlList = {"rtmp://live.hkstv.hk.lxdns.com/live/hks",
2017-12-10 01:34:43 +08:00
"rtmp://live.hkstv.hk.lxdns.com/live/hks"
2017-09-30 13:00:12 +08:00
//rtsp链接支持输入用户名密码
2017-12-10 01:34:43 +08:00
/*"rtsp://admin:jzan123456@192.168.0.122/"*/};
2017-05-02 23:43:10 +08:00
map<string , PlayerProxy::Ptr> proxyMap;
int i=0;
2017-09-30 13:00:12 +08:00
for(auto &url : urlList){
2017-05-05 21:29:44 +08:00
//PlayerProxy构造函数前两个参数分别为应用名app,流idstreamId
2017-08-09 18:39:30 +08:00
//比如说应用为live流id为0那么直播地址为:
2017-05-05 21:29:44 +08:00
//http://127.0.0.1/live/0/hls.m3u8
//rtsp://127.0.0.1/live/0
//rtmp://127.0.0.1/live/0
2017-09-30 13:00:12 +08:00
//录像地址为(当然vlc不支持这么多级的rtmp url可以用test_player测试rtmp点播):
2017-05-05 21:29:44 +08:00
//http://127.0.0.1/record/live/0/2017-04-11/11-09-38.mp4
//rtsp://127.0.0.1/record/live/0/2017-04-11/11-09-38.mp4
//rtmp://127.0.0.1/record/live/0/2017-04-11/11-09-38.mp4
2017-12-10 01:34:43 +08:00
PlayerProxy::Ptr player(new PlayerProxy("live",to_string(i).data()));
2017-09-30 13:00:12 +08:00
//指定RTP over TCP(播放rtsp时有效)
(*player)[RtspPlayer::kRtpType] = PlayerBase::RTP_TCP;
//开始播放,如果播放失败或者播放中止,将会自动重试若干次,重试次数在配置文件中配置,默认一直重试
2017-05-02 23:43:10 +08:00
player->play(url);
2017-09-30 13:00:12 +08:00
//需要保存PlayerProxy否则作用域结束就会销毁该对象
2017-12-10 01:34:43 +08:00
proxyMap.emplace(to_string(i),player);
++i;
2017-05-02 23:43:10 +08:00
}
2017-05-03 00:12:51 +08:00
#ifdef ENABLE_OPENSSL
2017-05-02 23:43:10 +08:00
//请把证书"test_server.pem"放置在本程序可执行程序同目录下
try{
2017-12-01 11:42:49 +08:00
//加载证书,证书包含公钥和私钥
2017-05-02 23:43:10 +08:00
SSL_Initor::Instance().loadServerPem((exePath() + ".pem").data());
}catch(...){
FatalL << "请把证书:" << (exeName() + ".pem") << "放置在本程序可执行程序同目录下:" << exeDir() << endl;
2017-09-30 13:00:12 +08:00
proxyMap.clear();
2017-05-02 23:43:10 +08:00
return 0;
}
2017-05-03 00:12:51 +08:00
#endif //ENABLE_OPENSSL
2017-04-05 09:39:16 +08:00
2017-09-30 13:00:12 +08:00
//简单的telnet服务器可用于服务器调试但是不能使用23端口否则telnet上了莫名其妙的现象
2017-08-10 16:16:42 +08:00
//测试方法:telnet 127.0.0.1 8023
//输入用户名和密码登录(user:test,pwd:123456)输入help命令查看帮助
TcpServer<ShellSession>::Ptr shellSrv(new TcpServer<ShellSession>());
2017-09-30 13:00:12 +08:00
ShellSession::addUser("test", "123456");
shellSrv->start(8023);
//开启rtsp/rtmp/http服务器
2017-04-05 09:39:16 +08:00
TcpServer<RtspSession>::Ptr rtspSrv(new TcpServer<RtspSession>());
TcpServer<RtmpSession>::Ptr rtmpSrv(new TcpServer<RtmpSession>());
TcpServer<HttpSession>::Ptr httpSrv(new TcpServer<HttpSession>());
2017-09-30 13:00:12 +08:00
rtspSrv->start(mINI::Instance()[Config::Rtsp::kPort]);//默认554
rtmpSrv->start(mINI::Instance()[Config::Rtmp::kPort]);//默认1935
httpSrv->start(mINI::Instance()[Config::Http::kPort]);//默认80
2017-05-03 00:12:51 +08:00
#ifdef ENABLE_OPENSSL
2017-09-30 13:00:12 +08:00
//如果支持ssl还可以开启https服务器
2017-05-03 00:12:51 +08:00
TcpServer<HttpsSession>::Ptr httpsSrv(new TcpServer<HttpsSession>());
2017-09-30 13:00:12 +08:00
httpsSrv->start(mINI::Instance()[Config::Http::kSSLPort]);//默认443
2017-05-03 00:12:51 +08:00
#endif //ENABLE_OPENSSL
2017-04-05 09:39:16 +08:00
EventPoller::Instance().runLoop();
2017-09-30 13:00:12 +08:00
//销毁拉流客户端
2017-05-05 21:29:44 +08:00
proxyMap.clear();
2017-09-30 13:00:12 +08:00
//销毁服务器
2017-08-10 16:16:42 +08:00
shellSrv.reset();
2017-04-05 09:39:16 +08:00
rtspSrv.reset();
rtmpSrv.reset();
httpSrv.reset();
2017-08-09 18:39:30 +08:00
2017-05-03 00:12:51 +08:00
#ifdef ENABLE_OPENSSL
2017-04-19 17:47:07 +08:00
httpsSrv.reset();
2017-05-03 00:12:51 +08:00
#endif //ENABLE_OPENSSL
2017-05-02 23:43:10 +08:00
2017-09-30 13:00:12 +08:00
//rtsp服务器用到udp端口分配器了
2017-05-02 23:43:10 +08:00
UDPServer::Destory();
2017-09-30 13:00:12 +08:00
//TcpServer用到了WorkThreadPool
2017-05-02 23:43:10 +08:00
WorkThreadPool::Destory();
2018-01-25 09:12:41 +08:00
AsyncTaskThread::Destory();
2017-05-02 23:43:10 +08:00
EventPoller::Destory();
Logger::Destory();
2017-04-05 09:39:16 +08:00
return 0;
}