ZLMediaKit/server/main.cpp

528 lines
21 KiB
C++
Raw Normal View History

2020-04-04 20:30:09 +08:00
/*
* Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
2019-05-20 11:22:59 +08:00
*
* This file is part of ZLMediaKit(https://github.com/xia-chu/ZLMediaKit).
2019-05-20 11:22:59 +08:00
*
2020-04-04 20:30:09 +08:00
* Use of this source code is governed by MIT license that can be found in the
* LICENSE file in the root of the source tree. All contributing project authors
* may be found in the AUTHORS file in the root of the source tree.
2019-05-20 11:22:59 +08:00
*/
#include <signal.h>
#include <iostream>
#include "Util/File.h"
#include "Util/logger.h"
#include "Util/SSLBox.h"
#include "Util/onceToken.h"
#include "Util/CMD.h"
#include "Network/TcpServer.h"
#include "Poller/EventPoller.h"
#include "Common/config.h"
#include "Rtsp/RtspSession.h"
#include "Rtmp/RtmpSession.h"
#include "Shell/ShellSession.h"
#include "Http/WebSocketSession.h"
#include "Rtp/RtpServer.h"
2019-06-06 18:28:33 +08:00
#include "WebApi.h"
2019-06-24 14:58:56 +08:00
#include "WebHook.h"
2021-03-31 17:15:26 +08:00
#include "../webrtc/Sdp.h"
2019-05-20 11:22:59 +08:00
#if defined(ENABLE_VERSION)
#include "Version.h"
#endif
2019-06-15 17:07:10 +08:00
#if !defined(_WIN32)
#include "System.h"
#endif//!defined(_WIN32)
2019-05-20 11:22:59 +08:00
using namespace std;
using namespace toolkit;
using namespace mediakit;
namespace mediakit {
////////////HTTP配置///////////
namespace Http {
#define HTTP_FIELD "http."
2019-06-19 10:32:54 +08:00
const string kPort = HTTP_FIELD"port";
const string kSSLPort = HTTP_FIELD"sslport";
2019-05-20 11:22:59 +08:00
onceToken token1([](){
2019-12-06 11:54:10 +08:00
mINI::Instance()[kPort] = 80;
mINI::Instance()[kSSLPort] = 443;
2019-05-20 11:22:59 +08:00
},nullptr);
}//namespace Http
////////////SHELL配置///////////
namespace Shell {
#define SHELL_FIELD "shell."
2019-06-19 10:32:54 +08:00
const string kPort = SHELL_FIELD"port";
2019-05-20 11:22:59 +08:00
onceToken token1([](){
2019-12-06 11:54:10 +08:00
mINI::Instance()[kPort] = 9000;
2019-05-20 11:22:59 +08:00
},nullptr);
} //namespace Shell
////////////RTSP服务器配置///////////
namespace Rtsp {
#define RTSP_FIELD "rtsp."
2019-06-19 10:32:54 +08:00
const string kPort = RTSP_FIELD"port";
const string kSSLPort = RTSP_FIELD"sslport";
2019-05-20 11:22:59 +08:00
onceToken token1([](){
2019-12-06 11:54:10 +08:00
mINI::Instance()[kPort] = 554;
mINI::Instance()[kSSLPort] = 332;
2019-05-20 11:22:59 +08:00
},nullptr);
} //namespace Rtsp
////////////RTMP服务器配置///////////
namespace Rtmp {
#define RTMP_FIELD "rtmp."
2019-06-19 10:32:54 +08:00
const string kPort = RTMP_FIELD"port";
2020-04-29 11:25:15 +08:00
const string kSSLPort = RTMP_FIELD"sslport";
2019-05-20 11:22:59 +08:00
onceToken token1([](){
2019-12-06 11:54:10 +08:00
mINI::Instance()[kPort] = 1935;
2020-04-29 11:25:15 +08:00
mINI::Instance()[kSSLPort] = 19350;
2019-05-20 11:22:59 +08:00
},nullptr);
} //namespace RTMP
2019-12-06 11:54:10 +08:00
////////////Rtp代理相关配置///////////
namespace RtpProxy {
#define RTP_PROXY_FIELD "rtp_proxy."
const string kPort = RTP_PROXY_FIELD"port";
onceToken token1([](){
mINI::Instance()[kPort] = 10000;
},nullptr);
} //namespace RtpProxy
2019-05-20 11:22:59 +08:00
} // namespace mediakit
class CMD_main : public CMD {
public:
CMD_main() {
_parser.reset(new OptionParser(nullptr));
2019-06-15 17:07:10 +08:00
#if !defined(_WIN32)
2019-05-20 11:22:59 +08:00
(*_parser) << Option('d',/*该选项简称,如果是\x00则说明无简称*/
"daemon",/*该选项全称,每个选项必须有全称不得为null或空字符串*/
Option::ArgNone,/*该选项后面必须跟值*/
nullptr,/*该选项默认值*/
false,/*该选项是否必须赋值如果没有默认值且为ArgRequired时用户必须提供该参数否则将抛异常*/
"是否以Daemon方式启动",/*该选项说明文字*/
nullptr);
2019-06-15 17:07:10 +08:00
#endif//!defined(_WIN32)
2019-05-20 11:22:59 +08:00
(*_parser) << Option('l',/*该选项简称,如果是\x00则说明无简称*/
"level",/*该选项全称,每个选项必须有全称不得为null或空字符串*/
Option::ArgRequired,/*该选项后面必须跟值*/
to_string(LTrace).data(),/*该选项默认值*/
false,/*该选项是否必须赋值如果没有默认值且为ArgRequired时用户必须提供该参数否则将抛异常*/
"日志等级,LTrace~LError(0~4)",/*该选项说明文字*/
nullptr);
2019-08-06 22:51:40 +08:00
(*_parser) << Option('m',/*该选项简称,如果是\x00则说明无简称*/
"max_day",/*该选项全称,每个选项必须有全称不得为null或空字符串*/
Option::ArgRequired,/*该选项后面必须跟值*/
"7",/*该选项默认值*/
false,/*该选项是否必须赋值如果没有默认值且为ArgRequired时用户必须提供该参数否则将抛异常*/
2019-08-06 22:56:09 +08:00
"日志最多保存天数",/*该选项说明文字*/
2019-08-06 22:51:40 +08:00
nullptr);
2019-05-20 11:22:59 +08:00
(*_parser) << Option('c',/*该选项简称,如果是\x00则说明无简称*/
"config",/*该选项全称,每个选项必须有全称不得为null或空字符串*/
Option::ArgRequired,/*该选项后面必须跟值*/
(exeDir() + "config.ini").data(),/*该选项默认值*/
false,/*该选项是否必须赋值如果没有默认值且为ArgRequired时用户必须提供该参数否则将抛异常*/
"配置文件路径",/*该选项说明文字*/
nullptr);
(*_parser) << Option('s',/*该选项简称,如果是\x00则说明无简称*/
"ssl",/*该选项全称,每个选项必须有全称不得为null或空字符串*/
Option::ArgRequired,/*该选项后面必须跟值*/
(exeDir() + "ssl.p12").data(),/*该选项默认值*/
false,/*该选项是否必须赋值如果没有默认值且为ArgRequired时用户必须提供该参数否则将抛异常*/
2019-11-08 15:20:54 +08:00
"ssl证书文件或文件夹,支持p12/pem类型",/*该选项说明文字*/
2019-05-20 11:22:59 +08:00
nullptr);
(*_parser) << Option('t',/*该选项简称,如果是\x00则说明无简称*/
"threads",/*该选项全称,每个选项必须有全称不得为null或空字符串*/
Option::ArgRequired,/*该选项后面必须跟值*/
to_string(thread::hardware_concurrency()).data(),/*该选项默认值*/
false,/*该选项是否必须赋值如果没有默认值且为ArgRequired时用户必须提供该参数否则将抛异常*/
"启动事件触发线程数",/*该选项说明文字*/
nullptr);
#if defined(ENABLE_VERSION)
(*_parser) << Option('v', "version", Option::ArgNone, nullptr, false, "显示版本号",
[](const std::shared_ptr<ostream> &stream, const string &arg) -> bool {
//版本信息
*stream << "编译日期: " << build_time << std::endl;
*stream << "当前git分支: " << branch_name << std::endl;
*stream << "当前git hash值: " << commit_hash << std::endl;
throw ExitException();
});
#endif
2019-05-20 11:22:59 +08:00
}
~CMD_main() override{}
const char *description() const override{
2019-05-20 11:22:59 +08:00
return "主程序命令参数";
}
};
2019-06-15 17:07:10 +08:00
#if !defined(_WIN32)
2019-05-20 11:22:59 +08:00
static void inline listen_shell_input(){
cout << "> 欢迎进入命令模式,你可以输入\"help\"命令获取帮助" << endl;
cout << "> " << std::flush;
2019-05-20 22:22:05 +08:00
signal(SIGTTOU,SIG_IGN);
signal(SIGTTIN,SIG_IGN);
2019-06-15 17:07:10 +08:00
2019-05-20 11:22:59 +08:00
SockUtil::setNoBlocked(STDIN_FILENO);
auto oninput = [](int event) {
if (event & Event_Read) {
char buf[1024];
int n = read(STDIN_FILENO, buf, sizeof(buf));
if (n > 0) {
buf[n] = '\0';
try {
CMDRegister::Instance()(buf);
cout << "> " << std::flush;
} catch (ExitException &ex) {
InfoL << "ExitException";
kill(getpid(), SIGINT);
} catch (std::exception &ex) {
cout << ex.what() << endl;
}
} else {
DebugL << get_uv_errmsg();
EventPollerPool::Instance().getFirstPoller()->delEvent(STDIN_FILENO);
}
}
if (event & Event_Error) {
WarnL << "Event_Error";
EventPollerPool::Instance().getFirstPoller()->delEvent(STDIN_FILENO);
}
};
EventPollerPool::Instance().getFirstPoller()->addEvent(STDIN_FILENO, Event_Read | Event_Error | Event_LT,oninput);
}
2019-06-15 17:07:10 +08:00
#endif//!defined(_WIN32)
//全局变量在WebApi中用于保存配置文件用
string g_ini_file;
2019-09-24 15:21:20 +08:00
2021-03-31 17:15:26 +08:00
void test_sdp(){
char str1[] = "v=0\n"
"o=- 380154348540553537 2 IN IP4 127.0.0.1\n"
"s=-\n"
"b=CT:1900\n"
"t=0 0\n"
"a=group:BUNDLE video\n"
"a=msid-semantic: WMS\n"
"m=video 9 RTP/SAVPF 96\n"
"c=IN IP4 0.0.0.0\n"
"a=rtcp:9 IN IP4 0.0.0.0\n"
"a=ice-ufrag:1ZFN\n"
"a=ice-pwd:70P3H0jPlGz1fiJl5XZfXMZH\n"
"a=ice-options:trickle\n"
"a=fingerprint:sha-256 3E:10:35:6B:9A:9E:B0:55:AC:2A:88:F5:74:C1:70:32:B5:8D:88:1D:37:B0:9C:69:A6:DD:07:10:73:27:1A:16\n"
"a=setup:active\n"
"a=mid:video\n"
"a=recvonly\n"
"a=rtcp-mux\n"
"a=rtpmap:96 H264/90000\n"
"a=fmtp:96 level-asymmetry-allowed=1;packetization-mode=0;profile-level-id=42e01f";
char str2[] = "v=0\n"
"o=- 2584450093346841581 2 IN IP4 127.0.0.1\n"
"s=-\n"
"t=0 0\n"
"a=group:BUNDLE audio video data\n"
"a=msid-semantic: WMS 616cfbb1-33a3-4d8c-8275-a199d6005549\n"
"m=audio 9 UDP/TLS/RTP/SAVPF 111 103 104 9 0 8 106 105 13 110 112 113 126\n"
"c=IN IP4 0.0.0.0\n"
"a=rtcp:9 IN IP4 0.0.0.0\n"
"a=ice-ufrag:sXJ3\n"
"a=ice-pwd:yEclOTrLg1gEubBFefOqtmyV\n"
"a=fingerprint:sha-256 22:14:B5:AF:66:12:C7:C7:8D:EF:4B:DE:40:25:ED:5D:8F:17:54:DD:88:33:C0:13:2E:FD:1A:FA:7E:7A:1B:79\n"
"a=setup:actpass\n"
"a=mid:audio\n"
"a=extmap:1/sendonly urn:ietf:params:rtp-hdrext:ssrc-audio-level\n"
"a=sendrecv\n"
"a=rtcp-mux\n"
"a=rtpmap:111 opus/48000/2\n"
"a=rtcp-fb:111 transport-cc\n"
"a=fmtp:111 minptime=10;useinbandfec=1\n"
"a=rtpmap:103 ISAC/16000\n"
"a=rtpmap:104 ISAC/32000\n"
"a=rtpmap:9 G722/8000\n"
"a=rtpmap:0 PCMU/8000\n"
"a=rtpmap:8 PCMA/8000\n"
"a=rtpmap:106 CN/32000\n"
"a=rtpmap:105 CN/16000\n"
"a=rtpmap:13 CN/8000\n"
"a=rtpmap:110 telephone-event/48000\n"
"a=rtpmap:112 telephone-event/32000\n"
"a=rtpmap:113 telephone-event/16000\n"
"a=rtpmap:126 telephone-event/8000\n"
"a=ssrc:120276603 cname:iSkJ2vn5cYYubTve\n"
"a=ssrc:120276603 msid:616cfbb1-33a3-4d8c-8275-a199d6005549 1da3d329-7399-4fe9-b20f-69606bebd363\n"
"a=ssrc:120276603 mslabel:616cfbb1-33a3-4d8c-8275-a199d6005549\n"
"a=ssrc:120276603 label:1da3d329-7399-4fe9-b20f-69606bebd363\n"
"m=video 9 UDP/TLS/RTP/SAVPF 96 98 100 102 127 97 99 101 125\n"
"c=IN IP4 0.0.0.0\n"
"a=rtcp:9 IN IP4 0.0.0.0\n"
"a=ice-ufrag:sXJ3\n"
"a=ice-pwd:yEclOTrLg1gEubBFefOqtmyV\n"
"a=fingerprint:sha-256 22:14:B5:AF:66:12:C7:C7:8D:EF:4B:DE:40:25:ED:5D:8F:17:54:DD:88:33:C0:13:2E:FD:1A:FA:7E:7A:1B:79\n"
"a=setup:actpass\n"
"a=mid:video\n"
"a=extmap:2 urn:ietf:params:rtp-hdrext:toffset\n"
"a=extmap:3 http://www.webrtc.org/experiments/rtp-hdrext/abs-send-time\n"
"a=extmap:4 urn:3gpp:video-orientation\n"
"a=extmap:5 http://www.ietf.org/id/draft-holmer-rmcat-transport-wide-cc-extensions-01\n"
"a=extmap:6 http://www.webrtc.org/experiments/rtp-hdrext/playout-delay\n"
"a=sendrecv\n"
"a=rtcp-mux\n"
"a=rtcp-rsize\n"
"a=rtpmap:96 VP8/90000\n"
"a=rtcp-fb:96 ccm fir\n"
"a=rtcp-fb:96 nack\n"
"a=rtcp-fb:96 nack pli\n"
"a=rtcp-fb:96 goog-remb\n"
"a=rtcp-fb:96 transport-cc\n"
"a=rtpmap:98 VP9/90000\n"
"a=rtcp-fb:98 ccm fir\n"
"a=rtcp-fb:98 nack\n"
"a=rtcp-fb:98 nack pli\n"
"a=rtcp-fb:98 goog-remb\n"
"a=rtcp-fb:98 transport-cc\n"
"a=rtpmap:100 H264/90000\n"
"a=rtcp-fb:100 ccm fir\n"
"a=rtcp-fb:100 nack\n"
"a=rtcp-fb:100 nack pli\n"
"a=rtcp-fb:100 goog-remb\n"
"a=rtcp-fb:100 transport-cc\n"
"a=fmtp:100 level-asymmetry-allowed=1;packetization-mode=1;profile-level-id=42e01f\n"
"a=rtpmap:102 red/90000\n"
"a=rtpmap:127 ulpfec/90000\n"
"a=rtpmap:97 rtx/90000\n"
"a=fmtp:97 apt=96\n"
"a=rtpmap:99 rtx/90000\n"
"a=fmtp:99 apt=98\n"
"a=rtpmap:101 rtx/90000\n"
"a=fmtp:101 apt=100\n"
"a=rtpmap:125 rtx/90000\n"
"a=fmtp:125 apt=102\n"
"a=ssrc-group:FID 2580761338 611523443\n"
"a=ssrc:2580761338 cname:iSkJ2vn5cYYubTve\n"
"a=ssrc:2580761338 msid:616cfbb1-33a3-4d8c-8275-a199d6005549 bf270496-a23e-47b5-b901-ef23096cd961\n"
"a=ssrc:2580761338 mslabel:616cfbb1-33a3-4d8c-8275-a199d6005549\n"
"a=ssrc:2580761338 label:bf270496-a23e-47b5-b901-ef23096cd961\n"
"a=ssrc:611523443 cname:iSkJ2vn5cYYubTve\n"
"a=ssrc:611523443 msid:616cfbb1-33a3-4d8c-8275-a199d6005549 bf270496-a23e-47b5-b901-ef23096cd961\n"
"a=ssrc:611523443 mslabel:616cfbb1-33a3-4d8c-8275-a199d6005549\n"
"a=ssrc:611523443 label:bf270496-a23e-47b5-b901-ef23096cd961\n"
"a=candidate:3575467457 1 udp 2113937151 10.15.83.23 57857 typ host generation 0 ufrag 6R0z network-cost 999\n"
"m=application 9 DTLS/SCTP 5000\n"
"c=IN IP4 0.0.0.0\n"
"a=ice-ufrag:sXJ3\n"
"a=ice-pwd:yEclOTrLg1gEubBFefOqtmyV\n"
"a=fingerprint:sha-256 22:14:B5:AF:66:12:C7:C7:8D:EF:4B:DE:40:25:ED:5D:8F:17:54:DD:88:33:C0:13:2E:FD:1A:FA:7E:7A:1B:79\n"
"a=setup:actpass\n"
"a=mid:data\n"
"a=sctpmap:5000 webrtc-datachannel 1024\n"
"a=sctp-port:5000";
RtcSessionSdp sdp1;
sdp1.parse(str1);
RtcSessionSdp sdp2;
sdp2.parse(str2);
for (auto media : sdp1.medias) {
InfoL << getRtpDirectionString(media.getDirection());
}
for (auto media : sdp2.medias) {
InfoL << getRtpDirectionString(media.getDirection());
}
InfoL << sdp1.toString();
InfoL << sdp2.toString();
RtcSession session1;
session1.loadFrom(str1);
RtcSession session2;
session2.loadFrom(str2);
DebugL << session1.toString();
DebugL << session2.toString();
}
2019-09-24 15:21:20 +08:00
int start_main(int argc,char *argv[]) {
2019-05-20 16:26:04 +08:00
{
CMD_main cmd_main;
try {
cmd_main.operator()(argc, argv);
} catch (ExitException &) {
return 0;
2019-05-20 16:26:04 +08:00
} catch (std::exception &ex) {
cout << ex.what() << endl;
return -1;
}
2019-05-20 11:22:59 +08:00
2019-05-20 16:26:04 +08:00
bool bDaemon = cmd_main.hasKey("daemon");
LogLevel logLevel = (LogLevel) cmd_main["level"].as<int>();
logLevel = MIN(MAX(logLevel, LTrace), LError);
g_ini_file = cmd_main["config"];
2019-05-20 16:26:04 +08:00
string ssl_file = cmd_main["ssl"];
int threads = cmd_main["threads"];
2019-05-20 11:22:59 +08:00
2019-05-20 16:26:04 +08:00
//设置日志
Logger::Instance().add(std::make_shared<ConsoleChannel>("ConsoleChannel", logLevel));
2019-09-24 15:21:20 +08:00
#ifndef ANDROID
2019-08-06 22:51:40 +08:00
auto fileChannel = std::make_shared<FileChannel>("FileChannel", exeDir() + "log/", logLevel);
//日志最多保存天数
fileChannel->setMaxDay(cmd_main["max_day"]);
Logger::Instance().add(fileChannel);
2019-09-24 15:21:20 +08:00
#endif//
2019-05-20 11:22:59 +08:00
2019-06-15 17:07:10 +08:00
#if !defined(_WIN32)
2019-11-29 11:27:54 +08:00
pid_t pid = getpid();
2019-05-20 16:26:04 +08:00
if (bDaemon) {
//启动守护进程
System::startDaemon();
}
2019-05-20 16:34:46 +08:00
//开启崩溃捕获等
System::systemSetup();
2019-06-15 17:07:10 +08:00
#endif//!defined(_WIN32)
2019-05-20 16:34:46 +08:00
2019-05-20 16:26:04 +08:00
//启动异步日志线程
Logger::Instance().setWriter(std::make_shared<AsyncLogWriter>());
//加载配置文件,如果配置文件不存在就创建一个
loadIniConfig(g_ini_file.data());
2019-05-20 16:26:04 +08:00
2019-11-08 15:20:54 +08:00
if(!File::is_dir(ssl_file.data())){
//不是文件夹,加载证书,证书包含公钥和私钥
SSL_Initor::Instance().loadCertificate(ssl_file.data());
}else{
//加载文件夹下的所有证书
File::scanDir(ssl_file,[](const string &path, bool isDir){
if(!isDir){
2019-11-08 15:26:46 +08:00
//最后的一个证书会当做默认证书(客户端ssl握手时未指定主机)
2019-11-08 15:20:54 +08:00
SSL_Initor::Instance().loadCertificate(path.data());
}
return true;
});
}
2019-05-20 16:26:04 +08:00
2021-03-31 17:15:26 +08:00
// test_sdp();
2019-05-20 16:26:04 +08:00
uint16_t shellPort = mINI::Instance()[Shell::kPort];
uint16_t rtspPort = mINI::Instance()[Rtsp::kPort];
uint16_t rtspsPort = mINI::Instance()[Rtsp::kSSLPort];
uint16_t rtmpPort = mINI::Instance()[Rtmp::kPort];
2020-04-29 11:25:15 +08:00
uint16_t rtmpsPort = mINI::Instance()[Rtmp::kSSLPort];
2019-05-20 16:26:04 +08:00
uint16_t httpPort = mINI::Instance()[Http::kPort];
uint16_t httpsPort = mINI::Instance()[Http::kSSLPort];
2020-04-29 11:25:15 +08:00
uint16_t rtpPort = mINI::Instance()[RtpProxy::kPort];
2019-05-20 16:26:04 +08:00
2019-05-20 16:34:46 +08:00
//设置poller线程数,该函数必须在使用ZLToolKit网络相关对象之前调用才能生效
2019-05-20 16:26:04 +08:00
EventPollerPool::setPoolSize(threads);
//简单的telnet服务器可用于服务器调试但是不能使用23端口否则telnet上了莫名其妙的现象
//测试方法:telnet 127.0.0.1 9000
TcpServer::Ptr shellSrv(new TcpServer());
2020-04-29 11:25:15 +08:00
//rtsp[s]服务器, 可用于诸如亚马逊echo show这样的设备访问
2019-05-20 16:26:04 +08:00
TcpServer::Ptr rtspSrv(new TcpServer());
2020-04-29 11:25:15 +08:00
TcpServer::Ptr rtspSSLSrv(new TcpServer());
//rtmp[s]服务器
2019-05-20 16:26:04 +08:00
TcpServer::Ptr rtmpSrv(new TcpServer());
2020-04-29 11:25:15 +08:00
TcpServer::Ptr rtmpsSrv(new TcpServer());
//http[s]服务器
2019-05-20 16:26:04 +08:00
TcpServer::Ptr httpSrv(new TcpServer());
TcpServer::Ptr httpsSrv(new TcpServer());
2019-11-29 11:27:54 +08:00
2019-12-06 11:54:10 +08:00
#if defined(ENABLE_RTPPROXY)
2020-04-29 11:25:15 +08:00
//GB28181 rtp推流端口支持UDP/TCP
RtpServer::Ptr rtpServer = std::make_shared<RtpServer>();
2019-12-06 11:54:10 +08:00
#endif//defined(ENABLE_RTPPROXY)
2019-11-29 11:27:54 +08:00
try {
//rtsp服务器端口默认554
2020-04-29 11:25:15 +08:00
if(rtspPort) { rtspSrv->start<RtspSession>(rtspPort); }
2019-11-29 11:27:54 +08:00
//rtsps服务器端口默认322
2020-04-29 11:25:15 +08:00
if(rtspsPort) { rtspSSLSrv->start<RtspSessionWithSSL>(rtspsPort); }
2019-11-29 11:27:54 +08:00
//rtmp服务器端口默认1935
2020-04-29 11:25:15 +08:00
if(rtmpPort) { rtmpSrv->start<RtmpSession>(rtmpPort); }
//rtmps服务器端口默认19350
if(rtmpsPort) { rtmpsSrv->start<RtmpSessionWithSSL>(rtmpsPort); }
2019-11-29 11:27:54 +08:00
//http服务器端口默认80
2020-04-29 11:25:15 +08:00
if(httpPort) { httpSrv->start<HttpSession>(httpPort); }
2019-11-29 11:27:54 +08:00
//https服务器端口默认443
2020-04-29 11:25:15 +08:00
if(httpsPort) { httpsSrv->start<HttpsSession>(httpsPort); }
2019-11-29 11:27:54 +08:00
//telnet远程调试服务器
2020-04-29 11:25:15 +08:00
if(shellPort) { shellSrv->start<ShellSession>(shellPort); }
2019-12-06 11:54:10 +08:00
#if defined(ENABLE_RTPPROXY)
//创建rtp服务器
if(rtpPort){ rtpServer->start(rtpPort); }
2019-12-06 11:54:10 +08:00
#endif//defined(ENABLE_RTPPROXY)
2019-11-29 11:27:54 +08:00
}catch (std::exception &ex){
WarnL << "端口占用或无权限:" << ex.what() << endl;
ErrorL << "程序启动失败,请修改配置文件中端口号后重试!" << endl;
sleep(1);
#if !defined(_WIN32)
if(pid != getpid()){
kill(pid,SIGINT);
}
#endif
return -1;
}
2019-05-20 16:26:04 +08:00
installWebApi();
InfoL << "已启动http api 接口";
installWebHook();
InfoL << "已启动http hook 接口";
2019-09-24 15:21:20 +08:00
#if !defined(_WIN32) && !defined(ANDROID)
2019-05-20 16:26:04 +08:00
if (!bDaemon) {
//交互式shell输入
listen_shell_input();
}
2019-06-15 17:07:10 +08:00
#endif
2019-05-20 11:22:59 +08:00
2019-05-20 16:26:04 +08:00
//设置退出信号处理函数
static semaphore sem;
signal(SIGINT, [](int) {
InfoL << "SIGINT:exit";
signal(SIGINT, SIG_IGN);// 设置退出信号
2019-05-20 16:26:04 +08:00
sem.post();
});// 设置退出信号
2019-06-15 17:07:10 +08:00
#if !defined(_WIN32)
signal(SIGHUP, [](int) { mediakit::loadIniConfig(g_ini_file.data()); });
2019-06-15 17:07:10 +08:00
#endif
2019-05-20 16:26:04 +08:00
sem.wait();
}
unInstallWebApi();
unInstallWebHook();
2019-05-28 09:50:05 +08:00
//休眠1秒再退出防止资源释放顺序错误
InfoL << "程序退出中,请等待...";
2019-05-28 09:50:05 +08:00
sleep(1);
InfoL << "程序退出完毕!";
2020-03-20 11:51:24 +08:00
return 0;
2019-05-20 11:22:59 +08:00
}
2019-09-24 15:21:20 +08:00
#ifndef DISABLE_MAIN
int main(int argc,char *argv[]) {
return start_main(argc,argv);
}
#endif //DISABLE_MAIN