RtspSession.h 8 KB
Newer Older
xiongziliang committed
1
/*
xiongziliang committed
2
 * Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
xiongziliang committed
3 4 5
 *
 * This file is part of ZLMediaKit(https://github.com/xiongziliang/ZLMediaKit).
 *
xiongziliang committed
6 7 8
 * 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.
xzl committed
9 10 11 12 13 14
 */

#ifndef SESSION_RTSPSESSION_H_
#define SESSION_RTSPSESSION_H_

#include <set>
xiongzilaing committed
15
#include <vector>
16
#include <unordered_set>
xzl committed
17
#include <unordered_map>
xiongziliang committed
18 19
#include "Util/util.h"
#include "Util/logger.h"
xiongziliang committed
20
#include "Common/config.h"
xiongziliang committed
21 22
#include "Network/TcpSession.h"
#include "Player/PlayerBase.h"
xiongziliang committed
23
#include "RtpMultiCaster.h"
xzl committed
24
#include "RtspMediaSource.h"
xiongziliang committed
25
#include "RtspSplitter.h"
xiongziliang committed
26
#include "RtpReceiver.h"
27
#include "RtspMediaSourceImp.h"
xiongziliang committed
28
#include "Common/Stamp.h"
xzl committed
29 30

using namespace std;
xiongziliang committed
31
using namespace toolkit;
32

xiongziliang committed
33
namespace mediakit {
xzl committed
34 35

class RtspSession;
36

37
class BufferRtp : public Buffer{
38 39 40 41 42
public:
    typedef std::shared_ptr<BufferRtp> Ptr;
    BufferRtp(const RtpPacket::Ptr & pkt,uint32_t offset = 0 ):_rtp(pkt),_offset(offset){}
    virtual ~BufferRtp(){}

43
    char *data() const override {
xiongziliang committed
44
        return (char *)_rtp->data() + _offset;
45 46
    }
    uint32_t size() const override {
xiongziliang committed
47
        return _rtp->size() - _offset;
48 49 50 51 52 53
    }
private:
    RtpPacket::Ptr _rtp;
    uint32_t _offset;
};

xiongziliang committed
54
class RtspSession: public TcpSession, public RtspSplitter, public RtpReceiver , public MediaSourceEvent{
xzl committed
55
public:
56 57
    typedef std::shared_ptr<RtspSession> Ptr;
    typedef std::function<void(const string &realm)> onGetRealm;
58 59
    //encrypted为true是则表明是md5加密的密码,否则是明文密码
    //在请求明文密码时如果提供md5密码者则会导致认证失败
60 61 62 63 64 65 66 67
    typedef std::function<void(bool encrypted,const string &pwd_or_md5)> onAuth;

    RtspSession(const Socket::Ptr &pSock);
    virtual ~RtspSession();
    ////TcpSession override////
    void onRecv(const Buffer::Ptr &pBuf) override;
    void onError(const SockException &err) override;
    void onManager() override;
xiongziliang committed
68
protected:
69
    //RtspSplitter override
xiongziliang committed
70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89
    /**
     * 收到完整的rtsp包回调,包括sdp等content数据
     * @param parser rtsp包
     */
    void onWholeRtspPacket(Parser &parser) override;

    /**
     * 收到rtp包回调
     * @param data
     * @param len
     */
    void onRtpPacket(const char *data,uint64_t len) override;

    /**
     * 从rtsp头中获取Content长度
     * @param parser
     * @return
     */
    int64_t getContentLength(Parser &parser) override;

90 91 92 93 94
    //RtpReceiver override
    void onRtpSorted(const RtpPacket::Ptr &rtppt, int trackidx) override;
    //MediaSourceEvent override
    bool close(MediaSource &sender,bool force) override ;
    int totalReaderCount(MediaSource &sender) override;
95

96
    //TcpSession override
97
    int send(const Buffer::Ptr &pkt) override;
98 99 100 101 102 103 104 105

    /**
     * 收到RTCP包回调
     * @param iTrackidx
     * @param track
     * @param pucData
     * @param uiLen
     */
106
    virtual void onRtcpPacket(int iTrackidx, SdpTrack::Ptr &track, unsigned char *pucData, unsigned int uiLen);
xzl committed
107
private:
108
    //处理options方法,获取服务器能力
xiongziliang committed
109
    void handleReq_Options(const Parser &parser);
110
    //处理describe方法,请求服务器rtsp sdp信息
xiongziliang committed
111
    void handleReq_Describe(const Parser &parser);
112
    //处理ANNOUNCE方法,请求推流,附带sdp
xiongziliang committed
113
    void handleReq_ANNOUNCE(const Parser &parser);
114
    //处理record方法,开始推流
xiongziliang committed
115
    void handleReq_RECORD(const Parser &parser);
116
    //处理setup方法,播放和推流协商rtp传输方式用
xiongziliang committed
117
    void handleReq_Setup(const Parser &parser);
118
    //处理play方法,开始或恢复播放
xiongziliang committed
119
    void handleReq_Play(const Parser &parser);
120
    //处理pause方法,暂停播放
xiongziliang committed
121
    void handleReq_Pause(const Parser &parser);
122
    //处理teardown方法,结束播放
xiongziliang committed
123
    void handleReq_Teardown(const Parser &parser);
124
    //处理Get方法,rtp over http才用到
xiongziliang committed
125
    void handleReq_Get(const Parser &parser);
126
    //处理Post方法,rtp over http才用到
xiongziliang committed
127
    void handleReq_Post(const Parser &parser);
128
    //处理SET_PARAMETER、GET_PARAMETER方法,一般用于心跳
xiongziliang committed
129 130
    void handleReq_SET_PARAMETER(const Parser &parser);

131 132 133 134 135 136 137 138 139 140 141 142 143
    //rtsp资源未找到
    void inline send_StreamNotFound();
    //不支持的传输模式
    void inline send_UnsupportedTransport();
    //会话id错误
    void inline send_SessionNotFound();
    //一般rtsp服务器打开端口失败时触发
    void inline send_NotAcceptable();
    //ssrc转字符串
    inline string printSSRC(uint32_t ui32Ssrc);

    //获取track下标
    inline int getTrackIndexByTrackType(TrackType type);
144
    inline int getTrackIndexByControlSuffix(const string &controlSuffix);
145
    inline int getTrackIndexByInterleaved(int interleaved);
146

147 148 149 150
    //一般用于接收udp打洞包,也用于rtsp推流
    inline void onRcvPeerUdpData(int intervaled, const Buffer::Ptr &pBuf, const struct sockaddr &addr);
    //配合onRcvPeerUdpData使用
    inline void startListenPeerUdpData(int iTrackIdx);
xzl committed
151

xiongziliang committed
152
    ////rtsp专有认证相关////
153
    //认证成功
154
    void onAuthSuccess();
155
    //认证失败
156
    void onAuthFailed(const string &realm,const string &why,bool close = true);
157
    //开始走rtsp专有认证流程
158
    void onAuthUser(const string &realm,const string &authorization);
159
    //校验base64方式的认证加密
160
    void onAuthBasic(const string &realm,const string &strBase64);
161
    //校验md5方式的认证加密
162
    void onAuthDigest(const string &realm,const string &strMd5);
163

164
    //发送rtp给客户端
165
    void sendRtpPacket(const RtspMediaSource::RingDataType &pkt);
166 167 168 169 170 171 172
    //回复客户端
    bool sendRtspResponse(const string &res_code,const std::initializer_list<string> &header, const string &sdp = "" , const char *protocol = "RTSP/1.0");
    bool sendRtspResponse(const string &res_code,const StrCaseMap &header = StrCaseMap(), const string &sdp = "",const char *protocol = "RTSP/1.0");
    //服务器发送rtcp
    void sendSenderReport(bool overTcp,int iTrackIndex);
    //设置socket标志
    void setSocketFlags();
xiongziliang committed
173
private:
174 175 176 177 178 179 180 181 182 183 184
    //用于判断客户端是否超时
    Ticker _ticker;
    //收到的seq,回复时一致
    int _iCseq = 0;
    //ContentBase
    string _strContentBase;
    //Session号
    string _strSession;
    //是否第一次播放,第一次播放需要鉴权,第二次播放属于暂停恢复
    bool _bFirstPlay = true;
    //url解析后保存的相关信息
185
    MediaInfo _mediaInfo;
186 187 188
    //rtsp播放器绑定的直播源
    std::weak_ptr<RtspMediaSource> _pMediaSrc;
    //直播源读取器
189
    RtspMediaSource::RingType::RingReader::Ptr _pRtpReader;
190 191 192 193 194 195 196 197 198 199
    //推流或拉流客户端采用的rtp传输方式
    Rtsp::eRtpType _rtpType = Rtsp::RTP_Invalid;
    //sdp里面有效的track,包含音频或视频
    vector<SdpTrack::Ptr> _aTrackInfo;
    ////////RTP over udp////////
    //RTP端口,trackid idx 为数组下标
    Socket::Ptr _apRtpSock[2];
    //RTCP端口,trackid idx 为数组下标
    Socket::Ptr _apRtcpSock[2];
    //标记是否收到播放的udp打洞包,收到播放的udp打洞包后才能知道其外网udp端口号
200
    unordered_set<int> _udpSockConnected;
201 202 203
    ////////RTP over udp_multicast////////
    //共享的rtp组播对象
    RtpMultiCaster::Ptr _multicaster;
xzl committed
204

205
    //登录认证
206
    string _strNonce;
207 208
    //消耗的总流量
    uint64_t _ui64TotalBytes = 0;
209

210 211 212 213 214 215
    //RTSP over HTTP
    //quicktime 请求rtsp会产生两次tcp连接,
    //一次发送 get 一次发送post,需要通过x-sessioncookie关联起来
    string _http_x_sessioncookie;
    function<void(const Buffer::Ptr &pBuf)> _onRecv;
    //是否开始发送rtp
216
    bool _enableSendRtp;
xiongziliang committed
217
    //rtsp推流相关
218 219 220 221 222
    RtspMediaSourceImp::Ptr _pushSrc;
    //rtcp统计,trackid idx 为数组下标
    RtcpCounter _aRtcpCnt[2];
    //rtcp发送时间,trackid idx 为数组下标
    Ticker _aRtcpTicker[2];
xzl committed
223 224
};

225 226 227 228 229
/**
 * 支持ssl加密的rtsp服务器,可用于诸如亚马逊echo show这样的设备访问
 */
typedef TcpSessionWithSSL<RtspSession> RtspSessionWithSSL;

xiongziliang committed
230
} /* namespace mediakit */
xzl committed
231 232

#endif /* SESSION_RTSPSESSION_H_ */