mirror of https://github.com/procxx/kepka.git
Use simple loader for non-streamable file types.
This commit is contained in:
parent
e1114530ab
commit
8704f6efd0
|
@ -1196,6 +1196,13 @@ bool DocumentData::hasRemoteLocation() const {
|
|||
return (_dc != 0 && _access != 0);
|
||||
}
|
||||
|
||||
bool DocumentData::useStreamingLoader() const {
|
||||
return isAnimation()
|
||||
|| isVideoFile()
|
||||
|| isAudioFile()
|
||||
|| isVoiceMessage();
|
||||
}
|
||||
|
||||
bool DocumentData::canBeStreamed() const {
|
||||
// For now video messages are not streamed.
|
||||
return hasRemoteLocation() && supportsStreaming() && !isVideoMessage();
|
||||
|
@ -1203,10 +1210,7 @@ bool DocumentData::canBeStreamed() const {
|
|||
|
||||
bool DocumentData::canBePlayed() const {
|
||||
return !_inappPlaybackFailed
|
||||
&& (isAnimation()
|
||||
|| isVideoFile()
|
||||
|| isAudioFile()
|
||||
|| isVoiceMessage())
|
||||
&& useStreamingLoader()
|
||||
&& (loaded() || canBeStreamed());
|
||||
}
|
||||
|
||||
|
@ -1220,6 +1224,9 @@ bool DocumentData::inappPlaybackFailed() const {
|
|||
|
||||
auto DocumentData::createStreamingLoader(Data::FileOrigin origin) const
|
||||
-> std::unique_ptr<Media::Streaming::Loader> {
|
||||
if (!useStreamingLoader()) {
|
||||
return nullptr;
|
||||
}
|
||||
const auto &location = this->location(true);
|
||||
if (!data().isEmpty()) {
|
||||
return Media::Streaming::MakeBytesLoader(data());
|
||||
|
|
|
@ -249,6 +249,7 @@ private:
|
|||
|
||||
void destroyLoader(FileLoader *newValue = nullptr) const;
|
||||
|
||||
[[nodiscard]] bool useStreamingLoader() const;
|
||||
[[nodiscard]] bool thumbnailEnoughForSticker() const;
|
||||
|
||||
// Two types of location: from MTProto by dc+access or from web by url
|
||||
|
|
|
@ -473,7 +473,7 @@ auto Reader::Slices::fill(int offset, bytes::span buffer) -> FillResult {
|
|||
if (_headerMode != HeaderMode::NoCache
|
||||
&& !(_header.flags & Flag::LoadedFromCache)) {
|
||||
// Waiting for initial cache query.
|
||||
Assert(_header.flags & Flag::LoadingFromCache);
|
||||
Assert(waitingForHeaderCache());
|
||||
return {};
|
||||
} else if (isFullInHeader()) {
|
||||
return fillFromHeader(offset, buffer);
|
||||
|
@ -587,16 +587,20 @@ QByteArray Reader::Slices::partForDownloader(int offset) const {
|
|||
return (i != end(slice.parts)) ? i->second : QByteArray();
|
||||
}
|
||||
|
||||
bool Reader::Slices::waitingForHeaderCache() const {
|
||||
return (_header.flags & Slice::Flag::LoadingFromCache);
|
||||
}
|
||||
|
||||
std::optional<int> Reader::Slices::readCacheRequiredFor(int offset) {
|
||||
Expects(offset < _size);
|
||||
Expects(!waitingForHeaderCache());
|
||||
|
||||
using Flag = Slice::Flag;
|
||||
if ((_header.flags & Flag::LoadingFromCache) || isFullInHeader()) {
|
||||
if (isFullInHeader()) {
|
||||
return std::nullopt;
|
||||
}
|
||||
const auto index = offset / kInSlice;
|
||||
auto &slice = _data[index];
|
||||
return (slice.flags & Flag::LoadedFromCache)
|
||||
return (slice.flags & Slice::Flag::LoadedFromCache)
|
||||
? std::nullopt
|
||||
: std::make_optional(index + 1);
|
||||
}
|
||||
|
@ -923,6 +927,9 @@ void Reader::processDownloaderRequests() {
|
|||
}
|
||||
|
||||
bool Reader::downloaderWaitForCachedSlice(int offset) {
|
||||
if (_slices.waitingForHeaderCache()) {
|
||||
return true;
|
||||
}
|
||||
const auto sliceNumber = _slices.readCacheRequiredFor(offset);
|
||||
if (sliceNumber.value_or(0) != _downloaderSliceNumber) {
|
||||
_downloaderSliceNumber = sliceNumber.value_or(0);
|
||||
|
|
|
@ -131,6 +131,7 @@ private:
|
|||
[[nodiscard]] bool headerModeUnknown() const;
|
||||
[[nodiscard]] bool isFullInHeader() const;
|
||||
[[nodiscard]] bool isGoodHeader() const;
|
||||
[[nodiscard]] bool waitingForHeaderCache() const;
|
||||
|
||||
void processCacheResult(int sliceNumber, PartsMap &&result);
|
||||
void processPart(int offset, QByteArray &&bytes);
|
||||
|
|
|
@ -47,8 +47,9 @@ StreamedFileDownloader::StreamedFileDownloader(
|
|||
, _origin(origin)
|
||||
, _cacheKey(cacheKey)
|
||||
, _fileLocationKey(fileLocationKey)
|
||||
, _reader(std::move(reader)) {
|
||||
_partIsSaved.resize((size + kPartSize - 1) / kPartSize, false);
|
||||
, _reader(std::move(reader))
|
||||
, _partsCount((size + kPartSize - 1) / kPartSize) {
|
||||
_partIsSaved.resize(_partsCount, false);
|
||||
|
||||
_reader->partsForDownloader(
|
||||
) | rpl::start_with_next([=](const LoadedPart &part) {
|
||||
|
@ -87,18 +88,19 @@ std::optional<MediaKey> StreamedFileDownloader::fileLocationKey() const {
|
|||
}
|
||||
|
||||
void StreamedFileDownloader::cancelRequests() {
|
||||
const auto requests = std::count(
|
||||
begin(_partIsSaved),
|
||||
begin(_partIsSaved) + _nextPartIndex,
|
||||
false);
|
||||
_queue->queriesCount -= requests;
|
||||
//_partsRequested == std::count(
|
||||
// begin(_partIsSaved),
|
||||
// begin(_partIsSaved) + _nextPartIndex,
|
||||
// false);
|
||||
_queue->queriesCount -= _partsRequested;
|
||||
_partsRequested = 0;
|
||||
_nextPartIndex = 0;
|
||||
|
||||
_reader->cancelForDownloader();
|
||||
}
|
||||
|
||||
bool StreamedFileDownloader::loadPart() {
|
||||
if (_finished || _nextPartIndex >= size(_partIsSaved)) {
|
||||
if (_finished || _nextPartIndex >= _partsCount) {
|
||||
return false;
|
||||
}
|
||||
const auto index = std::find(
|
||||
|
@ -106,17 +108,19 @@ bool StreamedFileDownloader::loadPart() {
|
|||
end(_partIsSaved),
|
||||
false
|
||||
) - begin(_partIsSaved);
|
||||
if (index == size(_partIsSaved)) {
|
||||
_nextPartIndex = index;
|
||||
if (index == _partsCount) {
|
||||
_nextPartIndex = _partsCount;
|
||||
return false;
|
||||
}
|
||||
_nextPartIndex = index + 1;
|
||||
_reader->loadForDownloader(index * kPartSize);
|
||||
|
||||
AssertIsDebug();
|
||||
//_downloader->requestedAmountIncrement(
|
||||
// requestData.dcId,
|
||||
// requestData.dcIndex,
|
||||
// kPartSize);
|
||||
++_partsRequested;
|
||||
++_queue->queriesCount;
|
||||
|
||||
return true;
|
||||
|
@ -132,11 +136,12 @@ void StreamedFileDownloader::savePart(const LoadedPart &part) {
|
|||
|
||||
const auto offset = part.offset;
|
||||
const auto index = offset / kPartSize;
|
||||
Assert(index >= 0 && index < _partIsSaved.size());
|
||||
Assert(index >= 0 && index < _partsCount);
|
||||
if (_partIsSaved[index]) {
|
||||
return;
|
||||
}
|
||||
_partIsSaved[index] = true;
|
||||
++_partsSaved;
|
||||
|
||||
if (index < _nextPartIndex) {
|
||||
AssertIsDebug();
|
||||
|
@ -144,12 +149,13 @@ void StreamedFileDownloader::savePart(const LoadedPart &part) {
|
|||
// requestData.dcId,
|
||||
// requestData.dcIndex,
|
||||
// -kPartSize);
|
||||
--_partsRequested;
|
||||
--_queue->queriesCount;
|
||||
}
|
||||
if (!writeResultPart(offset, bytes::make_span(part.bytes))) {
|
||||
return;
|
||||
}
|
||||
if (ranges::find(_partIsSaved, false) == end(_partIsSaved)) {
|
||||
if (_partsSaved == _partsCount) {
|
||||
if (!finalizeResult()) {
|
||||
return;
|
||||
}
|
||||
|
|
|
@ -59,6 +59,9 @@ private:
|
|||
|
||||
std::vector<bool> _partIsSaved; // vector<bool> :D
|
||||
int _nextPartIndex = 0;
|
||||
int _partsCount = 0;
|
||||
int _partsRequested = 0;
|
||||
int _partsSaved = 0;
|
||||
|
||||
rpl::lifetime _lifetime;
|
||||
|
||||
|
|
Loading…
Reference in New Issue