/* * 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 HLSRECORDER_H #define HLSRECORDER_H #include "HlsMakerImp.h" #include "MPEG.h" #include "MP4Muxer.h" #include "Common/config.h" namespace mediakit { template class HlsRecorderBase : public MediaSourceEventInterceptor, public Muxer, public std::enable_shared_from_this > { public: HlsRecorderBase(bool is_fmp4, const std::string &m3u8_file, const std::string ¶ms, const ProtocolOption &option) { GET_CONFIG(uint32_t, hlsNum, Hls::kSegmentNum); GET_CONFIG(bool, hlsKeep, Hls::kSegmentKeep); GET_CONFIG(uint32_t, hlsBufSize, Hls::kFileBufSize); GET_CONFIG(float, hlsDuration, Hls::kSegmentDuration); _option = option; _hls = std::make_shared(is_fmp4, m3u8_file, params, hlsBufSize, hlsDuration, hlsNum, hlsKeep); // 清空上次的残余文件 [AUTO-TRANSLATED:e16122be] // Clear the residual files from the last time _hls->clearCache(); } void setMediaSource(const MediaTuple& tuple) { _hls->setMediaSource(tuple); } void setListener(const std::weak_ptr &listener) { setDelegate(listener); _hls->getMediaSource()->setListener(this->shared_from_this()); } int readerCount() { return _hls->getMediaSource()->readerCount(); } void onReaderChanged(MediaSource &sender, int size) override { // hls保留切片个数为0时代表为hls录制(不删除切片),那么不管有无观看者都一直生成hls [AUTO-TRANSLATED:55709255] // When the number of hls slices is 0, it means hls recording (not deleting slices), so hls is generated all the time regardless of whether there are viewers _enabled = _option.hls_demand ? (_hls->isLive() ? size : true) : true; if (!size && _hls->isLive() && _option.hls_demand) { // hls直播时,如果无人观看就删除视频缓存,目的是为了防止视频跳跃 [AUTO-TRANSLATED:1d875c6a] // When hls is live, if no one is watching, delete the video cache to prevent video jumping _clear_cache = true; } MediaSourceEventInterceptor::onReaderChanged(sender, size); } bool inputFrame(const Frame::Ptr &frame) override { if (_clear_cache && _option.hls_demand) { _clear_cache = false; // 清空旧的m3u8索引文件于ts切片 [AUTO-TRANSLATED:a4ce0664] // Clear the old m3u8 index file and ts slices _hls->clearCache(); _hls->getMediaSource()->setIndexFile(""); } if (_enabled || !_option.hls_demand) { return Muxer::inputFrame(frame); } return false; } bool isEnabled() { // 缓存尚未清空时,还允许触发inputFrame函数,以便及时清空缓存 [AUTO-TRANSLATED:7cfd4d49] // When the cache has not been cleared, it is still allowed to trigger the inputFrame function to clear the cache in time return _option.hls_demand ? (_clear_cache ? true : _enabled) : true; } protected: bool _enabled = true; bool _clear_cache = false; ProtocolOption _option; std::shared_ptr _hls; }; class HlsRecorder final : public HlsRecorderBase { public: using Ptr = std::shared_ptr; template HlsRecorder(ARGS && ...args) : HlsRecorderBase(false, std::forward(args)...) {} ~HlsRecorder() override { try { this->flush(); } catch (std::exception &ex) { WarnL << ex.what(); } } private: void onWrite(std::shared_ptr buffer, uint64_t timestamp, bool key_pos) override { if (!buffer) { // reset tracks _hls->inputData(nullptr, 0, timestamp, key_pos); } else { _hls->inputData(buffer->data(), buffer->size(), timestamp, key_pos); } } }; class HlsFMP4Recorder final : public HlsRecorderBase { public: using Ptr = std::shared_ptr; template HlsFMP4Recorder(ARGS && ...args) : HlsRecorderBase(true, std::forward(args)...) {} ~HlsFMP4Recorder() override { try { this->flush(); } catch (std::exception &ex) { WarnL << ex.what(); } } void addTrackCompleted() override { HlsRecorderBase::addTrackCompleted(); auto data = getInitSegment(); _hls->inputInitSegment(data.data(), data.size()); } private: void onSegmentData(std::string buffer, uint64_t timestamp, bool key_pos) override { if (buffer.empty()) { // reset tracks _hls->inputData(nullptr, 0, timestamp, key_pos); } else { _hls->inputData((char *)buffer.data(), buffer.size(), timestamp, key_pos); } } }; }//namespace mediakit #endif //HLSRECORDER_H