143 lines
5.3 KiB
C++
143 lines
5.3 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 HLSRECORDER_H
|
||
#define HLSRECORDER_H
|
||
|
||
#include "HlsMakerImp.h"
|
||
#include "MPEG.h"
|
||
#include "MP4Muxer.h"
|
||
#include "Common/config.h"
|
||
|
||
namespace mediakit {
|
||
|
||
template <typename Muxer>
|
||
class HlsRecorderBase : public MediaSourceEventInterceptor, public Muxer, public std::enable_shared_from_this<HlsRecorderBase<Muxer> > {
|
||
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<HlsMakerImp>(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<MediaSourceEvent> &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<HlsMakerImp> _hls;
|
||
};
|
||
|
||
class HlsRecorder final : public HlsRecorderBase<MpegMuxer> {
|
||
public:
|
||
using Ptr = std::shared_ptr<HlsRecorder>;
|
||
template <typename ...ARGS>
|
||
HlsRecorder(ARGS && ...args) : HlsRecorderBase<MpegMuxer>(false, std::forward<ARGS>(args)...) {}
|
||
~HlsRecorder() override {
|
||
try {
|
||
this->flush();
|
||
} catch (std::exception &ex) {
|
||
WarnL << ex.what();
|
||
}
|
||
}
|
||
|
||
private:
|
||
void onWrite(std::shared_ptr<toolkit::Buffer> 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<MP4MuxerMemory> {
|
||
public:
|
||
using Ptr = std::shared_ptr<HlsFMP4Recorder>;
|
||
template <typename ...ARGS>
|
||
HlsFMP4Recorder(ARGS && ...args) : HlsRecorderBase<MP4MuxerMemory>(true, std::forward<ARGS>(args)...) {}
|
||
~HlsFMP4Recorder() override {
|
||
try {
|
||
this->flush();
|
||
} catch (std::exception &ex) {
|
||
WarnL << ex.what();
|
||
}
|
||
}
|
||
|
||
void addTrackCompleted() override {
|
||
HlsRecorderBase<MP4MuxerMemory>::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
|