tdesktop/Telegram/SourceFiles/storage/cache/storage_cache_binlog_reader...

112 lines
2.4 KiB
C++

/*
This file is part of Telegram Desktop,
the official desktop application for the Telegram messaging service.
For license and copyright information please follow this link:
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/
#include "storage/cache/storage_cache_binlog_reader.h"
namespace Storage {
namespace Cache {
namespace details {
BinlogWrapper::BinlogWrapper(
File &binlog,
const Settings &settings,
int64 till)
: _binlog(binlog)
, _settings(settings)
, _till(till ? till : _binlog.size())
, _data(_settings.readBlockSize)
, _full(_data) {
}
bool BinlogWrapper::finished() const {
return _finished;
}
bool BinlogWrapper::failed() const {
return _failed;
}
std::optional<BasicHeader> BinlogWrapper::ReadHeader(
File &binlog,
const Settings &settings) {
auto result = BasicHeader();
if (binlog.offset() != 0) {
return {};
} else if (binlog.read(bytes::object_as_span(&result)) != sizeof(result)) {
return {};
} else if (result.getFormat() != Format::Format_0) {
return {};
} else if (settings.trackEstimatedTime
!= !!(result.flags & result.kTrackEstimatedTime)) {
return {};
}
return result;
}
bool BinlogWrapper::readPart() {
if (_finished) {
return false;
}
const auto no = [&] {
finish();
return false;
};
const auto offset = _binlog.offset();
const auto left = (_till - offset);
if (!left) {
return no();
}
if (!_part.empty() && _full.data() != _part.data()) {
bytes::move(_full, _part);
_part = _full.subspan(0, _part.size());
}
const auto amount = std::min(
left,
int64(_full.size() - _part.size()));
Assert(amount > 0);
const auto readBytes = _binlog.read(
_full.subspan(_part.size(), amount));
if (!readBytes) {
return no();
}
_part = _full.subspan(0, _part.size() + readBytes);
return true;
}
bytes::const_span BinlogWrapper::readRecord(ReadRecordSize readRecordSize) {
if (_finished) {
return {};
}
const auto size = readRecordSize(*this, _part);
if (size == kRecordSizeUnknown || size > _part.size()) {
return {};
} else if (size == kRecordSizeInvalid) {
finish();
_finished = _failed = true;
return {};
}
Assert(size >= 0);
const auto result = _part.subspan(0, size);
_part = _part.subspan(size);
return result;
}
void BinlogWrapper::finish(size_type rollback) {
Expects(rollback >= 0);
if (rollback > 0) {
_failed = true;
}
rollback += _part.size();
_binlog.seek(_binlog.offset() - rollback);
}
} // namespace details
} // namespace Cache
} // namespace Storage