/* * Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved. * * This file is part of ZLMediaKit(https://github.com/xia-chu/ZLMediaKit). * * 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. */ #include "Common/config.h" #include "PlayerProxy.h" #include "Util/mini.h" #include "Util/MD5.h" #include "Util/logger.h" #include "Extension/AAC.h" using namespace toolkit; namespace mediakit { PlayerProxy::PlayerProxy(const string &vhost, const string &app, const string &stream_id, bool enable_hls, bool enable_mp4, int retry_count, const EventPoller::Ptr &poller) : MediaPlayer(poller) { _vhost = vhost; _app = app; _stream_id = stream_id; _enable_hls = enable_hls; _enable_mp4 = enable_mp4; _retry_count = retry_count; _on_close = [](const SockException &) {}; } void PlayerProxy::setPlayCallbackOnce(const function &cb) { _on_play = cb; } void PlayerProxy::setOnClose(const function &cb) { _on_close = cb ? cb : [](const SockException &) {}; } void PlayerProxy::play(const string &strUrlTmp) { weak_ptr weakSelf = shared_from_this(); std::shared_ptr piFailedCnt(new int(0)); //连续播放失败次数 setOnPlayResult([weakSelf, strUrlTmp, piFailedCnt](const SockException &err) { auto strongSelf = weakSelf.lock(); if (!strongSelf) { return; } if (strongSelf->_on_play) { strongSelf->_on_play(err); strongSelf->_on_play = nullptr; } if (!err) { // 播放成功 *piFailedCnt = 0;//连续播放失败次数清0 strongSelf->onPlaySuccess(); } else if (*piFailedCnt < strongSelf->_retry_count || strongSelf->_retry_count < 0) { // 播放失败,延时重试播放 strongSelf->rePlay(strUrlTmp, (*piFailedCnt)++); } else { //达到了最大重试次数,回调关闭 strongSelf->_on_close(err); } }); setOnShutdown([weakSelf, strUrlTmp, piFailedCnt](const SockException &err) { auto strongSelf = weakSelf.lock(); if (!strongSelf) { return; } //注销直接拉流代理产生的流:#532 strongSelf->setMediaSource(nullptr); if (strongSelf->_muxer) { auto tracks = strongSelf->MediaPlayer::getTracks(false); for (auto &track : tracks) { track->delDelegate(strongSelf->_muxer.get()); } GET_CONFIG(bool, resetWhenRePlay, General::kResetWhenRePlay); if (resetWhenRePlay) { strongSelf->_muxer.reset(); } else { strongSelf->_muxer->resetTracks(); } } //播放异常中断,延时重试播放 if (*piFailedCnt < strongSelf->_retry_count || strongSelf->_retry_count < 0) { strongSelf->rePlay(strUrlTmp, (*piFailedCnt)++); } else { //达到了最大重试次数,回调关闭 strongSelf->_on_close(err); } }); MediaPlayer::play(strUrlTmp); _pull_url = strUrlTmp; setDirectProxy(); } void PlayerProxy::setDirectProxy() { MediaSource::Ptr mediaSource; if (dynamic_pointer_cast(_delegate)) { //rtsp拉流 GET_CONFIG(bool, directProxy, Rtsp::kDirectProxy); if (directProxy) { mediaSource = std::make_shared(_vhost, _app, _stream_id); } } else if (dynamic_pointer_cast(_delegate)) { //rtmp拉流,rtmp强制直接代理 mediaSource = std::make_shared(_vhost, _app, _stream_id); } if (mediaSource) { setMediaSource(mediaSource); mediaSource->setListener(shared_from_this()); } } PlayerProxy::~PlayerProxy() { _timer.reset(); } void PlayerProxy::rePlay(const string &strUrl, int iFailedCnt) { auto iDelay = MAX(2 * 1000, MIN(iFailedCnt * 3000, 60 * 1000)); weak_ptr weakSelf = shared_from_this(); _timer = std::make_shared(iDelay / 1000.0f, [weakSelf, strUrl, iFailedCnt]() { //播放失败次数越多,则延时越长 auto strongPlayer = weakSelf.lock(); if (!strongPlayer) { return false; } WarnL << "重试播放[" << iFailedCnt << "]:" << strUrl; strongPlayer->MediaPlayer::play(strUrl); strongPlayer->setDirectProxy(); return false; }, getPoller()); } bool PlayerProxy::close(MediaSource &sender, bool force) { if (!force && totalReaderCount()) { return false; } //通知其停止推流 weak_ptr weakSelf = dynamic_pointer_cast(shared_from_this()); getPoller()->async_first([weakSelf]() { auto strongSelf = weakSelf.lock(); if (!strongSelf) { return; } strongSelf->_muxer.reset(); strongSelf->setMediaSource(nullptr); strongSelf->teardown(); }); _on_close(SockException(Err_shutdown, "closed by user")); WarnL << sender.getSchema() << "/" << sender.getVhost() << "/" << sender.getApp() << "/" << sender.getId() << " " << force; return true; } int PlayerProxy::totalReaderCount() { return (_muxer ? _muxer->totalReaderCount() : 0) + (_pMediaSrc ? _pMediaSrc->readerCount() : 0); } int PlayerProxy::totalReaderCount(MediaSource &sender) { return totalReaderCount(); } MediaOriginType PlayerProxy::getOriginType(MediaSource &sender) const { return MediaOriginType::pull; } string PlayerProxy::getOriginUrl(MediaSource &sender) const { return _pull_url; } std::shared_ptr PlayerProxy::getOriginSock(MediaSource &sender) const { return getSockInfo(); } void PlayerProxy::onPlaySuccess() { GET_CONFIG(bool, resetWhenRePlay, General::kResetWhenRePlay); if (dynamic_pointer_cast(_pMediaSrc)) { //rtsp拉流代理 if (resetWhenRePlay || !_muxer) { _muxer.reset(new MultiMediaSourceMuxer(_vhost, _app, _stream_id, getDuration(), false, true, _enable_hls, _enable_mp4)); } } else if (dynamic_pointer_cast(_pMediaSrc)) { //rtmp拉流代理 if (resetWhenRePlay || !_muxer) { _muxer.reset(new MultiMediaSourceMuxer(_vhost, _app, _stream_id, getDuration(), true, false, _enable_hls, _enable_mp4)); } } else { //其他拉流代理 if (resetWhenRePlay || !_muxer) { _muxer.reset(new MultiMediaSourceMuxer(_vhost, _app, _stream_id, getDuration(), true, true, _enable_hls, _enable_mp4)); } } _muxer->setMediaListener(shared_from_this()); auto videoTrack = getTrack(TrackVideo, false); if (videoTrack) { //添加视频 _muxer->addTrack(videoTrack); //视频数据写入_mediaMuxer videoTrack->addDelegate(_muxer); } auto audioTrack = getTrack(TrackAudio, false); if (audioTrack) { //添加音频 _muxer->addTrack(audioTrack); //音频数据写入_mediaMuxer audioTrack->addDelegate(_muxer); } //添加完毕所有track,防止单track情况下最大等待3秒 _muxer->addTrackCompleted(); if (_pMediaSrc) { //让_muxer对象拦截一部分事件(比如说录像相关事件) _pMediaSrc->setListener(_muxer); } } } /* namespace mediakit */