ZLMediaKit/src/Rtsp/Rtsp.cpp

300 lines
7.4 KiB
C++
Raw Normal View History

2017-10-09 22:11:01 +08:00
/*
2017-09-27 16:20:30 +08:00
* MIT License
2017-04-01 16:35:56 +08:00
*
2019-05-08 15:40:07 +08:00
* Copyright (c) 2016-2019 xiongziliang <771730766@qq.com>
2017-09-27 16:20:30 +08:00
*
* 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-01 16:35:56 +08:00
*/
#include <stdlib.h>
2017-04-25 11:35:41 +08:00
#include "Rtsp.h"
2019-06-28 16:48:02 +08:00
#include "Common/Parser.h"
2017-04-25 11:35:41 +08:00
2019-03-27 18:41:52 +08:00
namespace mediakit{
2019-08-15 19:13:31 +08:00
static void getAttrSdp(const map<string, string> &attr, _StrPrinter &printer){
const map<string, string>::value_type *ptr = nullptr;
for(auto &pr : attr){
if(pr.first == "control"){
ptr = &pr;
continue;
}
if(pr.second.empty()){
printer << "a=" << pr.first << "\r\n";
}else{
printer << "a=" << pr.first << ":" << pr.second << "\r\n";
}
}
if(ptr){
printer << "a=" << ptr->first << ":" << ptr->second << "\r\n";
}
}
string SdpTrack::toString() const {
_StrPrinter _printer;
switch (_type){
case TrackTitle:{
_printer << "v=" << 0 << "\r\n";
if(!_o.empty()){
_printer << "o="<< _o << "\r\n";
}
if(!_c.empty()){
_printer << "c=" << _c << "\r\n";
}
if(!_t.empty()){
_printer << "t=" << _t << "\r\n";
}
_printer << "s=RTSP Session, streamed by the ZLMediaKit\r\n";
_printer << "i=ZLMediaKit Live Stream\r\n";
getAttrSdp(_attr,_printer);
}
break;
case TrackAudio:
case TrackVideo:{
if(_type == TrackAudio){
_printer << "m=audio 0 RTP/AVP " << _pt << "\r\n";
}else{
_printer << "m=video 0 RTP/AVP " << _pt << "\r\n";
}
if(!_b.empty()){
_printer << "b=" <<_b << "\r\n";
}
getAttrSdp(_attr,_printer);
}
break;
default:
break;
}
return _printer;
}
2019-06-28 16:48:02 +08:00
void SdpParser::load(const string &sdp) {
2018-10-25 22:57:59 +08:00
_track_map.clear();
string key;
SdpTrack::Ptr track = std::make_shared<SdpTrack>();
2018-10-25 18:50:18 +08:00
auto lines = split(sdp,"\n");
for (auto &line : lines){
trim(line);
2018-10-25 22:57:59 +08:00
if(line.size() < 2 || line[1] != '='){
2018-10-25 18:50:18 +08:00
continue;
}
char opt = line[0];
string opt_val = line.substr(2);
switch (opt){
case 'o':
2018-10-25 22:57:59 +08:00
track->_o = opt_val;
break;
2018-10-25 18:50:18 +08:00
case 's':
2018-10-25 22:57:59 +08:00
track->_s = opt_val;
break;
2018-10-25 18:50:18 +08:00
case 'i':
2018-10-25 22:57:59 +08:00
track->_i = opt_val;
break;
2018-10-25 18:50:18 +08:00
case 'c':
2018-10-25 22:57:59 +08:00
track->_c = opt_val;
break;
case 't':
track->_t = opt_val;
break;
case 'b':
track->_b = opt_val;
2018-10-25 18:50:18 +08:00
break;
case 'm':{
2018-10-25 22:57:59 +08:00
_track_map[key] = track;
track = std::make_shared<SdpTrack>();
2019-06-28 16:48:02 +08:00
key = FindField(opt_val.data(), nullptr," ");
2018-10-25 22:57:59 +08:00
track->_m = opt_val;
2018-10-25 18:50:18 +08:00
}
break;
case 'a':{
string attr = FindField(opt_val.data(), nullptr,":");
if(attr.empty()){
2018-10-25 22:57:59 +08:00
track->_attr[opt_val] = "";
2018-10-25 18:50:18 +08:00
}else{
2018-10-25 22:57:59 +08:00
track->_attr[attr] = FindField(opt_val.data(),":", nullptr);
2018-10-25 18:50:18 +08:00
}
}
break;
default:
2018-10-25 22:57:59 +08:00
track->_other[opt] = opt_val;
2018-10-25 18:50:18 +08:00
break;
}
}
2018-10-25 22:57:59 +08:00
_track_map[key] = track;
2018-10-25 18:50:18 +08:00
2018-10-25 22:57:59 +08:00
for (auto &pr : _track_map) {
auto &track = *pr.second;
if (pr.first == "") {
2018-10-26 10:59:13 +08:00
track._type = TrackTitle;
2018-10-25 22:57:59 +08:00
} else if (pr.first == "video") {
2018-10-26 10:59:13 +08:00
track._type = TrackVideo;
2018-10-25 18:50:18 +08:00
} else if (pr.first == "audio") {
2018-10-26 10:59:13 +08:00
track._type = TrackAudio;
2018-10-25 22:57:59 +08:00
} else {
2018-10-26 10:59:13 +08:00
track._type = TrackInvalid;
2018-10-25 22:57:59 +08:00
}
2018-10-25 18:50:18 +08:00
2018-10-25 22:57:59 +08:00
auto it = track._attr.find("range");
if (it != track._attr.end()) {
char name[16] = {0}, start[16] = {0}, end[16] = {0};
int ret = sscanf(it->second.data(), "%15[^=]=%15[^-]-%15s", name, start, end);
if (3 == ret || 2 == ret) {
if (strcmp(start, "now") == 0) {
strcpy(start, "0");
}
track._start = atof(start);
track._end = atof(end);
track._duration = track._end - track._start;
2018-10-25 18:50:18 +08:00
}
}
2018-10-25 22:57:59 +08:00
it = track._attr.find("rtpmap");
if(it != track._attr.end()){
auto rtpmap = it->second;
int pt, samplerate;
char codec[16] = {0};
if (3 == sscanf(rtpmap.data(), "%d %15[^/]/%d", &pt, codec, &samplerate)) {
track._pt = pt;
track._codec = codec;
track._samplerate = samplerate;
}
}
it = track._attr.find("fmtp");
if(it != track._attr.end()) {
track._fmtp = it->second;
}
2018-10-25 18:50:18 +08:00
2018-10-25 22:57:59 +08:00
it = track._attr.find("control");
if(it != track._attr.end()) {
track._control = it->second;
2018-10-26 09:56:29 +08:00
auto surffix = string("/") + track._control;
track._control_surffix = surffix.substr(1 + surffix.rfind('/'));
2018-10-25 18:50:18 +08:00
}
}
2018-10-25 22:57:59 +08:00
}
2018-10-25 18:50:18 +08:00
2019-06-28 16:48:02 +08:00
bool SdpParser::available() const {
2018-10-25 23:24:23 +08:00
return getTrack(TrackAudio) || getTrack(TrackVideo);
}
2019-06-28 16:48:02 +08:00
SdpTrack::Ptr SdpParser::getTrack(TrackType type) const {
2018-10-25 22:57:59 +08:00
for (auto &pr : _track_map){
2018-10-26 10:59:13 +08:00
if(pr.second->_type == type){
2018-10-25 22:57:59 +08:00
return pr.second;
}
}
return nullptr;
}
2018-10-25 18:50:18 +08:00
2019-06-28 16:48:02 +08:00
vector<SdpTrack::Ptr> SdpParser::getAvailableTrack() const {
2018-10-26 09:56:29 +08:00
vector<SdpTrack::Ptr> ret;
auto video = getTrack(TrackVideo);
if(video){
ret.emplace_back(video);
2017-04-01 16:35:56 +08:00
}
2018-10-26 09:56:29 +08:00
auto audio = getTrack(TrackAudio);
if(audio){
ret.emplace_back(audio);
}
return ret;
2017-04-01 16:35:56 +08:00
}
2018-10-26 09:56:29 +08:00
2019-08-15 19:13:31 +08:00
string SdpParser::toString() const {
string title,audio,video;
for(auto &pr : _track_map){
switch (pr.second->_type){
case TrackTitle:{
title = pr.second->toString();
}
break;
case TrackVideo:{
video = pr.second->toString();
}
break;
case TrackAudio:{
audio = pr.second->toString();
}
break;
default:
break;
}
}
return title + video + audio;
}
bool RtspUrl::parse(const string &strUrl) {
auto schema = FindField(strUrl.data(), nullptr, "://");
bool isSSL = strcasecmp(schema.data(), "rtsps") == 0;
//查找"://"与"/"之间的字符串,用于提取用户名密码
auto middle_url = FindField(strUrl.data(), "://", "/");
if (middle_url.empty()) {
middle_url = FindField(strUrl.data(), "://", nullptr);
}
auto pos = middle_url.rfind('@');
if (pos == string::npos) {
//并没有用户名密码
return setup(isSSL, strUrl, "", "");
}
//包含用户名密码
auto user_pwd = middle_url.substr(0, pos);
auto suffix = strUrl.substr(schema.size() + 3 + pos + 1);
auto url = StrPrinter << "rtsp://" << suffix << endl;
if (user_pwd.find(":") == string::npos) {
return setup(isSSL, url, user_pwd, "");
}
auto user = FindField(user_pwd.data(), nullptr, ":");
auto pwd = FindField(user_pwd.data(), ":", nullptr);
return setup(isSSL, url, user, pwd);
}
bool RtspUrl::setup(bool isSSL, const string &strUrl, const string &strUser, const string &strPwd) {
auto ip = FindField(strUrl.data(), "://", "/");
if (ip.empty()) {
ip = split(FindField(strUrl.data(), "://", NULL), "?")[0];
}
auto port = atoi(FindField(ip.data(), ":", NULL).data());
if (port <= 0 || port > UINT16_MAX) {
//rtsp 默认端口554
port = isSSL ? 322 : 554;
} else {
//服务器域名
ip = FindField(ip.data(), NULL, ":");
}
if (ip.empty()) {
return false;
}
_url = std::move(strUrl);
_user = std::move(strUser);
_passwd = std::move(strPwd);
_host = std::move(ip);
_port = port;
_is_ssl = isSSL;
return true;
}
2019-03-27 18:41:52 +08:00
}//namespace mediakit
2018-10-25 22:57:59 +08:00