285 lines
14 KiB
C++
285 lines
14 KiB
C++
/*
|
||
* Copyright (c) 2016-present The ZLMediaKit project authors. All Rights Reserved.
|
||
*
|
||
* This file is part of ZLMediaKit(https://github.com/ZLMediaKit/ZLMediaKit).
|
||
*
|
||
* Use of this source code is governed by MIT-like 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.
|
||
*/
|
||
|
||
#ifndef SESSION_RTSPSESSION_H_
|
||
#define SESSION_RTSPSESSION_H_
|
||
|
||
#include <set>
|
||
#include <vector>
|
||
#include <unordered_set>
|
||
#include "Network/Session.h"
|
||
#include "RtspSplitter.h"
|
||
#include "RtpReceiver.h"
|
||
#include "Rtcp/RtcpContext.h"
|
||
#include "RtspMediaSource.h"
|
||
#include "RtspMediaSourceImp.h"
|
||
#include "RtpMultiCaster.h"
|
||
|
||
namespace mediakit {
|
||
|
||
using BufferRtp = toolkit::BufferOffset<toolkit::Buffer::Ptr>;
|
||
class RtspSession : public toolkit::Session, public RtspSplitter, public RtpReceiver, public MediaSourceEvent {
|
||
public:
|
||
using Ptr = std::shared_ptr<RtspSession>;
|
||
using onGetRealm = std::function<void(const std::string &realm)>;
|
||
// encrypted为true是则表明是md5加密的密码,否则是明文密码 [AUTO-TRANSLATED:cad96e51]
|
||
// `encrypted` being `true` indicates an MD5 encrypted password, otherwise it is a plain text password
|
||
// 在请求明文密码时如果提供md5密码者则会导致认证失败 [AUTO-TRANSLATED:8a38bff8]
|
||
// When requesting a plain text password, providing an MD5 password will result in authentication failure
|
||
using onAuth = std::function<void(bool encrypted, const std::string &pwd_or_md5)>;
|
||
|
||
RtspSession(const toolkit::Socket::Ptr &sock);
|
||
////Session override////
|
||
void onRecv(const toolkit::Buffer::Ptr &buf) override;
|
||
void onError(const toolkit::SockException &err) override;
|
||
void onManager() override;
|
||
|
||
protected:
|
||
/////RtspSplitter override/////
|
||
// 收到完整的rtsp包回调,包括sdp等content数据 [AUTO-TRANSLATED:efbe20df]
|
||
// Callback for receiving a complete RTSP packet, including SDP and other content data
|
||
void onWholeRtspPacket(Parser &parser) override;
|
||
// 收到rtp包回调 [AUTO-TRANSLATED:119f1cca]
|
||
// Callback for receiving an RTP packet
|
||
void onRtpPacket(const char *data, size_t len) override;
|
||
// 从rtsp头中获取Content长度 [AUTO-TRANSLATED:0e6f033e]
|
||
// Get the Content length from the RTSP header
|
||
ssize_t getContentLength(Parser &parser) override;
|
||
|
||
////RtpReceiver override////
|
||
void onRtpSorted(RtpPacket::Ptr rtp, int track_idx) override;
|
||
void onBeforeRtpSorted(const RtpPacket::Ptr &rtp, int track_index) override;
|
||
|
||
///////MediaSourceEvent override///////
|
||
// 关闭 [AUTO-TRANSLATED:92392f02]
|
||
// Close
|
||
bool close(MediaSource &sender) override;
|
||
// 播放总人数 [AUTO-TRANSLATED:c42a3161]
|
||
// Total number of players
|
||
int totalReaderCount(MediaSource &sender) override;
|
||
// 获取媒体源类型 [AUTO-TRANSLATED:34290a69]
|
||
// Get the media source type
|
||
MediaOriginType getOriginType(MediaSource &sender) const override;
|
||
// 获取媒体源url或者文件路径 [AUTO-TRANSLATED:fa34d795]
|
||
// Get the media source URL or file path
|
||
std::string getOriginUrl(MediaSource &sender) const override;
|
||
// 获取媒体源客户端相关信息 [AUTO-TRANSLATED:037ef910]
|
||
// Get the media source client related information
|
||
std::shared_ptr<SockInfo> getOriginSock(MediaSource &sender) const override;
|
||
// 由于支持断连续推,存在OwnerPoller变更的可能 [AUTO-TRANSLATED:1c863b40]
|
||
// Due to support for continuous pushing, there is a possibility of OwnerPoller changes
|
||
toolkit::EventPoller::Ptr getOwnerPoller(MediaSource &sender) override;
|
||
|
||
/////Session override////
|
||
ssize_t send(toolkit::Buffer::Ptr pkt) override;
|
||
// 收到RTCP包回调 [AUTO-TRANSLATED:249f4807]
|
||
// Callback for receiving an RTCP packet
|
||
virtual void onRtcpPacket(int track_idx, SdpTrack::Ptr &track, const char *data, size_t len);
|
||
|
||
// 回复客户端 [AUTO-TRANSLATED:8108ebea]
|
||
// Reply to the client
|
||
virtual bool sendRtspResponse(const std::string &res_code, const StrCaseMap &header = StrCaseMap(), const std::string &sdp = "", const char *protocol = "RTSP/1.0");
|
||
|
||
protected:
|
||
// url解析后保存的相关信息 [AUTO-TRANSLATED:1c26e4e3]
|
||
// Information related to the URL after parsing
|
||
MediaInfo _media_info;
|
||
|
||
////////RTP over udp_multicast////////
|
||
// 共享的rtp组播对象 [AUTO-TRANSLATED:d4a5cfdd]
|
||
// Shared RTP multicast object
|
||
RtpMultiCaster::Ptr _multicaster;
|
||
|
||
// Session号 [AUTO-TRANSLATED:4552ec74]
|
||
// Session number
|
||
std::string _sessionid;
|
||
|
||
uint32_t _multicast_ip = 0;
|
||
uint16_t _multicast_video_port = 0;
|
||
uint16_t _multicast_audio_port = 0;
|
||
|
||
private:
|
||
// 处理options方法,获取服务器能力 [AUTO-TRANSLATED:a51f6d7c]
|
||
// Handle the OPTIONS method, get server capabilities
|
||
void handleReq_Options(const Parser &parser);
|
||
// 处理describe方法,请求服务器rtsp sdp信息 [AUTO-TRANSLATED:ed2c8fcb]
|
||
// Handle the DESCRIBE method, request server RTSP SDP information
|
||
void handleReq_Describe(const Parser &parser);
|
||
// 处理ANNOUNCE方法,请求推流,附带sdp [AUTO-TRANSLATED:aa4b4517]
|
||
// Handle the ANNOUNCE method, request streaming, with SDP attached
|
||
void handleReq_ANNOUNCE(const Parser &parser);
|
||
// 处理record方法,开始推流 [AUTO-TRANSLATED:885cf8a9]
|
||
// Handle the RECORD method, start streaming
|
||
void handleReq_RECORD(const Parser &parser);
|
||
// 处理setup方法,播放和推流协商rtp传输方式用 [AUTO-TRANSLATED:cbe5dcfc]
|
||
// Handle the SETUP method, used for negotiating RTP transport methods for playback and streaming
|
||
void handleReq_Setup(const Parser &parser);
|
||
// 处理play方法,开始或恢复播放 [AUTO-TRANSLATED:f15d151d]
|
||
// Handle the PLAY method, start or resume playback
|
||
void handleReq_Play(const Parser &parser);
|
||
// 处理pause方法,暂停播放 [AUTO-TRANSLATED:0c3b8f79]
|
||
// Handle the PAUSE method, pause playback
|
||
void handleReq_Pause(const Parser &parser);
|
||
// 处理teardown方法,结束播放 [AUTO-TRANSLATED:64d82572]
|
||
// Handle the TEARDOWN method, end playback
|
||
void handleReq_Teardown(const Parser &parser);
|
||
// 处理Get方法,rtp over http才用到 [AUTO-TRANSLATED:c7c51eb6]
|
||
// Handle the GET method, only used for RTP over HTTP
|
||
void handleReq_Get(const Parser &parser);
|
||
// 处理Post方法,rtp over http才用到 [AUTO-TRANSLATED:228bdbbe]
|
||
// Handle the POST method, only used for RTP over HTTP
|
||
void handleReq_Post(const Parser &parser);
|
||
// 处理SET_PARAMETER、GET_PARAMETER方法,一般用于心跳 [AUTO-TRANSLATED:b9e333e1]
|
||
// Handle the SET_PARAMETER, GET_PARAMETER methods, generally used for heartbeats
|
||
void handleReq_SET_PARAMETER(const Parser &parser);
|
||
// rtsp资源未找到 [AUTO-TRANSLATED:9b779890]
|
||
// RTSP resource not found
|
||
void send_StreamNotFound();
|
||
// 不支持的传输模式 [AUTO-TRANSLATED:ef90414c]
|
||
// Unsupported transport mode
|
||
void send_UnsupportedTransport();
|
||
// 会话id错误 [AUTO-TRANSLATED:7cf632d3]
|
||
// Session ID error
|
||
void send_SessionNotFound();
|
||
// 一般rtsp服务器打开端口失败时触发 [AUTO-TRANSLATED:82ecb043]
|
||
// Triggered when the general RTSP server fails to open the port
|
||
void send_NotAcceptable();
|
||
// 获取track下标 [AUTO-TRANSLATED:36d0b2c2]
|
||
// Get the track index
|
||
int getTrackIndexByTrackType(TrackType type);
|
||
int getTrackIndexByControlUrl(const std::string &control_url);
|
||
int getTrackIndexByInterleaved(int interleaved);
|
||
// 一般用于接收udp打洞包,也用于rtsp推流 [AUTO-TRANSLATED:0b55c12f]
|
||
// Generally used to receive UDP hole punching packets, also used for RTSP pushing
|
||
void onRcvPeerUdpData(int interleaved, const toolkit::Buffer::Ptr &buf, const struct sockaddr_storage &addr);
|
||
// 配合onRcvPeerUdpData使用 [AUTO-TRANSLATED:811d2d1a]
|
||
// Used in conjunction with onRcvPeerUdpData
|
||
void startListenPeerUdpData(int track_idx);
|
||
// //rtsp专有认证相关//// [AUTO-TRANSLATED:0f021bb5]
|
||
// // RTSP specific authentication related ////
|
||
// 认证成功 [AUTO-TRANSLATED:e1bafff3]
|
||
// Authentication successful
|
||
void onAuthSuccess();
|
||
// 认证失败 [AUTO-TRANSLATED:a188326a]
|
||
// Authentication failed
|
||
void onAuthFailed(const std::string &realm, const std::string &why, bool close = true);
|
||
// 开始走rtsp专有认证流程 [AUTO-TRANSLATED:2d773497]
|
||
// Start the RTSP specific authentication process
|
||
void onAuthUser(const std::string &realm, const std::string &authorization);
|
||
// 校验base64方式的认证加密 [AUTO-TRANSLATED:bde8662f]
|
||
// Verify base64 authentication encryption
|
||
void onAuthBasic(const std::string &realm, const std::string &auth_base64);
|
||
// 校验md5方式的认证加密 [AUTO-TRANSLATED:0cc37fa7]
|
||
// Verify MD5 authentication encryption
|
||
void onAuthDigest(const std::string &realm, const std::string &auth_md5);
|
||
// 触发url鉴权事件 [AUTO-TRANSLATED:776dc4b5]
|
||
// Trigger URL authentication event
|
||
void emitOnPlay();
|
||
// 发送rtp给客户端 [AUTO-TRANSLATED:18602be0]
|
||
// Send RTP to the client
|
||
void sendRtpPacket(const RtspMediaSource::RingDataType &pkt);
|
||
// 触发rtcp发送 [AUTO-TRANSLATED:4fbe7706]
|
||
// Trigger RTCP sending
|
||
void updateRtcpContext(const RtpPacket::Ptr &rtp);
|
||
// 回复客户端 [AUTO-TRANSLATED:8108ebea]
|
||
// Reply to the client
|
||
bool sendRtspResponse(const std::string &res_code, const std::initializer_list<std::string> &header, const std::string &sdp = "", const char *protocol = "RTSP/1.0");
|
||
|
||
// 设置socket标志 [AUTO-TRANSLATED:4086e686]
|
||
// Set socket flag
|
||
void setSocketFlags();
|
||
|
||
private:
|
||
// 是否已经触发on_play事件 [AUTO-TRANSLATED:49c937ce]
|
||
// Whether the on_play event has been triggered
|
||
bool _emit_on_play = false;
|
||
bool _send_sr_rtcp[2] = {true, true};
|
||
// 断连续推延时 [AUTO-TRANSLATED:13ad578a]
|
||
// Delay in continuous pushing
|
||
uint32_t _continue_push_ms = 0;
|
||
// 推流或拉流客户端采用的rtp传输方式 [AUTO-TRANSLATED:27411079]
|
||
// RTP transport method used by the pushing or pulling client
|
||
Rtsp::eRtpType _rtp_type = Rtsp::RTP_Invalid;
|
||
// 收到的seq,回复时一致 [AUTO-TRANSLATED:64544fb4]
|
||
// Received seq, consistent when replying
|
||
int _cseq = 0;
|
||
// 消耗的总流量 [AUTO-TRANSLATED:45ad2785]
|
||
// Total traffic consumed
|
||
uint64_t _bytes_usage = 0;
|
||
//ContentBase
|
||
std::string _content_base;
|
||
// 记录是否需要rtsp专属鉴权,防止重复触发事件 [AUTO-TRANSLATED:9cff90b9]
|
||
// Record whether RTSP specific authentication is required to prevent duplicate event triggering
|
||
std::string _rtsp_realm;
|
||
// 登录认证 [AUTO-TRANSLATED:43fdb875]
|
||
// Login authentication
|
||
std::string _auth_nonce;
|
||
// 用于判断客户端是否超时 [AUTO-TRANSLATED:86cb328a]
|
||
// Used to determine if the client has timed out
|
||
toolkit::Ticker _alive_ticker;
|
||
|
||
// rtsp推流相关绑定的源 [AUTO-TRANSLATED:a25078d9]
|
||
// Source bound to RTSP pushing
|
||
RtspMediaSourceImp::Ptr _push_src;
|
||
// 推流器所有权 [AUTO-TRANSLATED:e47b4bcb]
|
||
// Pusher ownership
|
||
std::shared_ptr<void> _push_src_ownership;
|
||
// rtsp播放器绑定的直播源 [AUTO-TRANSLATED:a7e130b5]
|
||
// Live source bound to the RTSP player
|
||
std::weak_ptr<RtspMediaSource> _play_src;
|
||
// 直播源读取器 [AUTO-TRANSLATED:e1edc193]
|
||
// Live source reader
|
||
RtspMediaSource::RingType::RingReader::Ptr _play_reader;
|
||
// sdp里面有效的track,包含音频或视频 [AUTO-TRANSLATED:64e2fcdf]
|
||
// Valid track in SDP, including audio or video
|
||
std::vector<SdpTrack::Ptr> _sdp_track;
|
||
// 播放器setup指定的播放track,默认为TrackInvalid表示不指定即音视频都推 [AUTO-TRANSLATED:c7a0df5e]
|
||
// Track specified by the player setup, default is TrackInvalid, which means no specification, both audio and video are pushed
|
||
TrackType _target_play_track = TrackInvalid;
|
||
|
||
////////RTP over udp////////
|
||
// RTP端口,trackid idx 为数组下标 [AUTO-TRANSLATED:77c186bb]
|
||
// RTP port, trackid idx is the array index
|
||
toolkit::Socket::Ptr _rtp_socks[2];
|
||
// RTCP端口,trackid idx 为数组下标 [AUTO-TRANSLATED:446a7861]
|
||
// RTCP port, trackid idx is the array index
|
||
toolkit::Socket::Ptr _rtcp_socks[2];
|
||
// 标记是否收到播放的udp打洞包,收到播放的udp打洞包后才能知道其外网udp端口号 [AUTO-TRANSLATED:ad039c25]
|
||
// Flag whether the UDP hole punching packet for playback has been received. The external UDP port number can only be known after receiving the UDP hole punching packet for playback.
|
||
std::unordered_set<int> _udp_connected_flags;
|
||
////////RTSP over HTTP ////////
|
||
// quicktime 请求rtsp会产生两次tcp连接, [AUTO-TRANSLATED:3f72e181]
|
||
// QuickTime requests for RTSP will generate two TCP connections,
|
||
// 一次发送 get 一次发送post,需要通过x-sessioncookie关联起来 [AUTO-TRANSLATED:f29a653f]
|
||
// one for sending GET and one for sending POST. They need to be associated through x-sessioncookie.
|
||
std::string _http_x_sessioncookie;
|
||
std::function<void(const toolkit::Buffer::Ptr &)> _on_recv;
|
||
////////// rtcp ////////////////
|
||
// rtcp发送时间,trackid idx 为数组下标 [AUTO-TRANSLATED:bf3248b1]
|
||
// RTCP send time, trackid idx is the array index
|
||
toolkit::Ticker _rtcp_send_tickers[2];
|
||
// 统计rtp并发送rtcp [AUTO-TRANSLATED:0ac2b665]
|
||
// Count RTP and send RTCP
|
||
std::vector<RtcpContext::Ptr> _rtcp_context;
|
||
};
|
||
|
||
/**
|
||
* 支持ssl加密的rtsp服务器,可用于诸如亚马逊echo show这样的设备访问
|
||
* RTSP server supporting SSL encryption, which can be used for devices such as Amazon Echo Show to access.
|
||
|
||
|
||
* [AUTO-TRANSLATED:7d1eed83]
|
||
*/
|
||
using RtspSessionWithSSL = toolkit::SessionWithSSL<RtspSession>;
|
||
|
||
} /* namespace mediakit */
|
||
|
||
#endif /* SESSION_RTSPSESSION_H_ */
|