This commit is contained in:
kssenii 2022-02-01 22:10:56 +03:00
parent 218a633fec
commit 5cba11428a
5 changed files with 192 additions and 132 deletions

View File

@ -57,6 +57,12 @@ String FileSegment::getOrSetDownloader()
if (downloader_id.empty()) if (downloader_id.empty())
{ {
if (download_state != State::EMPTY
&& download_state != State::PARTIALLY_DOWNLOADED)
throw Exception(ErrorCodes::FILE_CACHE_ERROR,
"Can set downloader only for file segment with state EMPTY or PARTIALLY_DOWNLOADED, but got: {}",
download_state);
downloader_id = getCallerId(); downloader_id = getCallerId();
LOG_TEST(log, "Set downloader: {}, prev state: {}", downloader_id, stateToString(download_state)); LOG_TEST(log, "Set downloader: {}, prev state: {}", downloader_id, stateToString(download_state));
download_state = State::DOWNLOADING; download_state = State::DOWNLOADING;
@ -77,6 +83,7 @@ String FileSegment::getDownloader() const
bool FileSegment::isDownloader() const bool FileSegment::isDownloader() const
{ {
std::lock_guard segment_lock(mutex); std::lock_guard segment_lock(mutex);
LOG_TEST(log, "Checking for current downloader. Caller: {}, downloader: {}, current state: {}", getCallerId(), downloader_id, stateToString(download_state));
return getCallerId() == downloader_id; return getCallerId() == downloader_id;
} }
@ -121,6 +128,8 @@ void FileSegment::write(const char * from, size_t size)
} }
cache_writer->write(from, size); cache_writer->write(from, size);
cache_writer->next();
downloaded_size += size; downloaded_size += size;
} }
@ -135,7 +144,8 @@ FileSegment::State FileSegment::wait()
{ {
LOG_TEST(log, "{} waiting on: {}, current downloader: {}", getCallerId(), range().toString(), downloader_id); LOG_TEST(log, "{} waiting on: {}, current downloader: {}", getCallerId(), range().toString(), downloader_id);
assert(!downloader_id.empty() && downloader_id != getCallerId()); assert(!downloader_id.empty());
assert(downloader_id != getCallerId());
#ifndef NDEBUG #ifndef NDEBUG
{ {
@ -192,6 +202,8 @@ void FileSegment::completeBatchAndResetDownloader()
std::lock_guard segment_lock(mutex); std::lock_guard segment_lock(mutex);
bool is_downloader = downloader_id == getCallerId(); bool is_downloader = downloader_id == getCallerId();
std::cerr << "caller id: " << getCallerId() << "\n";
std::cerr << "downloader id: " << downloader_id << "\n";
if (!is_downloader) if (!is_downloader)
{ {
cv.notify_all(); cv.notify_all();
@ -214,7 +226,7 @@ void FileSegment::completeBatchAndResetDownloader()
cv.notify_all(); cv.notify_all();
} }
void FileSegment::complete(State state) void FileSegment::complete(State state, bool error)
{ {
{ {
std::lock_guard segment_lock(mutex); std::lock_guard segment_lock(mutex);
@ -226,6 +238,10 @@ void FileSegment::complete(State state)
throw Exception(ErrorCodes::FILE_CACHE_ERROR, throw Exception(ErrorCodes::FILE_CACHE_ERROR,
"File segment can be completed only by downloader or downloader's FileSegmentsHodler"); "File segment can be completed only by downloader or downloader's FileSegmentsHodler");
} }
else if (error)
{
remote_file_reader.reset();
}
if (state != State::DOWNLOADED if (state != State::DOWNLOADED
&& state != State::PARTIALLY_DOWNLOADED && state != State::PARTIALLY_DOWNLOADED

View File

@ -113,7 +113,7 @@ public:
void completeBatchAndResetDownloader(); void completeBatchAndResetDownloader();
void complete(State state); void complete(State state, bool error = false);
private: private:
size_t availableSize() const { return reserved_size - downloaded_size; } size_t availableSize() const { return reserved_size - downloaded_size; }

View File

@ -64,7 +64,7 @@ SeekableReadBufferPtr CachedReadBufferFromRemoteFS::getRemoteFSReadBuffer(FileSe
{ {
switch (read_type_) switch (read_type_)
{ {
case ReadType::REMOTE_FS_AND_PUT_IN_CACHE: case ReadType::REMOTE_FS_READ_AND_PUT_IN_CACHE:
{ {
/** /**
* Implementation (s3, hdfs, web) buffer might be passed through file segments. * Implementation (s3, hdfs, web) buffer might be passed through file segments.
@ -106,6 +106,8 @@ SeekableReadBufferPtr CachedReadBufferFromRemoteFS::getRemoteFSReadBuffer(FileSe
SeekableReadBufferPtr CachedReadBufferFromRemoteFS::getReadBufferForFileSegment(FileSegmentPtr file_segment) SeekableReadBufferPtr CachedReadBufferFromRemoteFS::getReadBufferForFileSegment(FileSegmentPtr file_segment)
{ {
assert(!file_segment->isDownloader());
auto range = file_segment->range(); auto range = file_segment->range();
[[maybe_unused]] bool first_segment_read_in_range = impl == nullptr; [[maybe_unused]] bool first_segment_read_in_range = impl == nullptr;
bytes_to_predownload = 0; bytes_to_predownload = 0;
@ -132,7 +134,7 @@ SeekableReadBufferPtr CachedReadBufferFromRemoteFS::getReadBufferForFileSegment(
auto downloader_id = file_segment->getOrSetDownloader(); auto downloader_id = file_segment->getOrSetDownloader();
if (downloader_id == file_segment->getCallerId()) if (downloader_id == file_segment->getCallerId())
{ {
read_type = ReadType::REMOTE_FS_AND_PUT_IN_CACHE; read_type = ReadType::REMOTE_FS_READ_AND_PUT_IN_CACHE;
implementation_buffer = getRemoteFSReadBuffer(file_segment, read_type); implementation_buffer = getRemoteFSReadBuffer(file_segment, read_type);
break; break;
@ -176,9 +178,6 @@ SeekableReadBufferPtr CachedReadBufferFromRemoteFS::getReadBufferForFileSegment(
break; break;
} }
case FileSegment::State::PARTIALLY_DOWNLOADED: case FileSegment::State::PARTIALLY_DOWNLOADED:
{
auto downloader_id = file_segment->getOrSetDownloader();
if (downloader_id == file_segment->getCallerId())
{ {
size_t download_offset = file_segment->downloadOffset(); size_t download_offset = file_segment->downloadOffset();
bool can_start_from_cache = download_offset && download_offset >= file_offset_of_buffer_end; bool can_start_from_cache = download_offset && download_offset >= file_offset_of_buffer_end;
@ -195,11 +194,9 @@ SeekableReadBufferPtr CachedReadBufferFromRemoteFS::getReadBufferForFileSegment(
read_type = ReadType::CACHED; read_type = ReadType::CACHED;
implementation_buffer = getCacheReadBuffer(range.left); implementation_buffer = getCacheReadBuffer(range.left);
break;
} }
else
{
read_type = ReadType::REMOTE_FS_AND_PUT_IN_CACHE;
implementation_buffer = getRemoteFSReadBuffer(file_segment, read_type);
LOG_TEST(log, "Current download offset: {}, file offset of buffer end: {}", download_offset, file_offset_of_buffer_end); LOG_TEST(log, "Current download offset: {}, file offset of buffer end: {}", download_offset, file_offset_of_buffer_end);
if (download_offset && download_offset + 1 < file_offset_of_buffer_end) if (download_offset && download_offset + 1 < file_offset_of_buffer_end)
@ -212,26 +209,35 @@ SeekableReadBufferPtr CachedReadBufferFromRemoteFS::getReadBufferForFileSegment(
/// ^ /// ^
/// file_offset_of_buffer_end /// file_offset_of_buffer_end
assert(first_segment_read_in_range); read_type = ReadType::REMOTE_FS_READ_BYPASS_CACHE;
bytes_to_predownload = file_offset_of_buffer_end - file_segment->downloadOffset() - 1; implementation_buffer = getRemoteFSReadBuffer(file_segment, read_type);
LOG_TEST(log, "Bytes to predownload {} for {}", bytes_to_predownload, downloader_id); // assert(first_segment_read_in_range);
} // bytes_to_predownload = file_offset_of_buffer_end - file_segment->downloadOffset() - 1;
}
break; break;
} }
else
auto downloader_id = file_segment->getOrSetDownloader();
if (downloader_id == file_segment->getCallerId())
{ {
read_type = ReadType::REMOTE_FS_READ_AND_PUT_IN_CACHE;
implementation_buffer = getRemoteFSReadBuffer(file_segment, read_type);
break;
}
download_state = FileSegment::State::DOWNLOADING; download_state = FileSegment::State::DOWNLOADING;
continue; continue;
} }
} }
}
break; break;
} }
[[maybe_unused]] auto download_current_segment = read_type == ReadType::REMOTE_FS_READ_AND_PUT_IN_CACHE;
assert(download_current_segment == file_segment->isDownloader());
assert(file_segment->range() == range); assert(file_segment->range() == range);
assert(file_offset_of_buffer_end >= range.left && file_offset_of_buffer_end <= range.right); assert(file_offset_of_buffer_end >= range.left && file_offset_of_buffer_end <= range.right);
@ -250,8 +256,13 @@ SeekableReadBufferPtr CachedReadBufferFromRemoteFS::getReadBufferForFileSegment(
implementation_buffer->seek(seek_offset, SEEK_SET); implementation_buffer->seek(seek_offset, SEEK_SET);
auto * file_reader = dynamic_cast<ReadBufferFromFile *>(implementation_buffer.get()); auto * file_reader = dynamic_cast<ReadBufferFromFile *>(implementation_buffer.get());
LOG_TEST(log, "Cache file: {}. Cached seek to: {}, file size: {}", size_t file_size = file_reader->size();
file_reader->getFileName(), seek_offset, file_reader->size()); auto state = file_segment->state();
LOG_TEST(log, "Cache file: {}. Cached seek to: {}, file size: {}, file segment state: {}, download offset: {}",
file_reader->getFileName(), seek_offset, file_size, state, file_segment->downloadOffset());
assert(file_size > 0);
break; break;
} }
case ReadType::REMOTE_FS_READ_BYPASS_CACHE: case ReadType::REMOTE_FS_READ_BYPASS_CACHE:
@ -259,8 +270,10 @@ SeekableReadBufferPtr CachedReadBufferFromRemoteFS::getReadBufferForFileSegment(
implementation_buffer->seek(file_offset_of_buffer_end, SEEK_SET); implementation_buffer->seek(file_offset_of_buffer_end, SEEK_SET);
break; break;
} }
case ReadType::REMOTE_FS_AND_PUT_IN_CACHE: case ReadType::REMOTE_FS_READ_AND_PUT_IN_CACHE:
{ {
assert(file_segment->isDownloader());
if (bytes_to_predownload) if (bytes_to_predownload)
{ {
size_t download_offset = file_segment->downloadOffset(); size_t download_offset = file_segment->downloadOffset();
@ -271,11 +284,13 @@ SeekableReadBufferPtr CachedReadBufferFromRemoteFS::getReadBufferForFileSegment(
} }
else else
{ {
assert(!first_segment_read_in_range || file_offset_of_buffer_end == range.left); LOG_TEST(log, "Impl buffer seek to : {}, range: {}, download_offset: {}",
implementation_buffer->seek(file_offset_of_buffer_end, SEEK_SET); file_offset_of_buffer_end, range.toString(), file_segment->downloadOffset());
LOG_TEST(log, "Impl buffer seek to current offset: {}", file_offset_of_buffer_end); // assert(!first_segment_read_in_range || file_offset_of_buffer_end == range.left);
implementation_buffer->seek(file_offset_of_buffer_end, SEEK_SET);
} }
break; break;
} }
} }
@ -338,7 +353,9 @@ bool CachedReadBufferFromRemoteFS::checkForPartialDownload()
if (file_offset_of_buffer_end > last_downloaded_offset) if (file_offset_of_buffer_end > last_downloaded_offset)
{ {
if (file_segment->state() == FileSegment::State::PARTIALLY_DOWNLOADED) if (file_segment->state() == FileSegment::State::PARTIALLY_DOWNLOADED)
read_type = ReadType::REMOTE_FS_AND_PUT_IN_CACHE; {
impl = getReadBufferForFileSegment(*current_file_segment_it);
}
else else
{ {
read_type = ReadType::REMOTE_FS_READ_BYPASS_CACHE; read_type = ReadType::REMOTE_FS_READ_BYPASS_CACHE;
@ -349,30 +366,11 @@ bool CachedReadBufferFromRemoteFS::checkForPartialDownload()
LOG_TEST( LOG_TEST(
log, "Changing read buffer from cache to remote filesystem read for file segment: {}, starting from offset: {}", log, "Changing read buffer from cache to remote filesystem read for file segment: {}, starting from offset: {}",
file_segment->range().toString(), file_offset_of_buffer_end); file_segment->range().toString(), file_offset_of_buffer_end);
}
return true; return true;
} }
} }
}
if (read_type == ReadType::REMOTE_FS_AND_PUT_IN_CACHE)
{
/**
* ReadType::REMOTE_FS_AND_PUT_IN_CACHE means that on previous getReadBufferForFileSegment() call
* current buffer successfully called file_segment->getOrSetDownloader() and became a downloader
* for this file segment. However, the downloader's term has a lifespan of 1 nextImpl() call,
* e.g. downloader reads buffer_size byte and calls completeBatchAndResetDownloader() and some other
* thread can become a downloader if it calls getOrSetDownloader() faster.
*
* So downloader is committed to download only buffer_size bytes and then is not a downloader anymore,
* because there is no guarantee on a higher level, that current buffer will not disappear without
* being destructed till the end of query or without finishing the read range, which he was supposed
* to read by marks range given to him. Therefore, each nextImpl() call, in case of
* READ_AND_PUT_IN_CACHE, starts with getOrSetDownloader().
*/
impl = getReadBufferForFileSegment(*current_file_segment_it);
return true;
}
return false; return false;
} }
@ -389,6 +387,11 @@ bool CachedReadBufferFromRemoteFS::nextImpl()
if (impl) if (impl)
{ {
{
auto & file_segment = *current_file_segment_it;
LOG_TEST(log, "Prev init. current read type: {}, range: {}, state: {}", toString(read_type), file_segment->range().toString(), file_segment->state());
}
if (!use_external_buffer) if (!use_external_buffer)
{ {
impl->position() = position(); impl->position() = position();
@ -400,21 +403,44 @@ bool CachedReadBufferFromRemoteFS::nextImpl()
assert(current_read_range.left <= file_offset_of_buffer_end); assert(current_read_range.left <= file_offset_of_buffer_end);
bool new_buf = false;
if (file_offset_of_buffer_end > current_read_range.right) if (file_offset_of_buffer_end > current_read_range.right)
{ {
if (!completeFileSegmentAndGetNext()) new_buf = completeFileSegmentAndGetNext();
if (!new_buf)
return false; return false;
} }
if (current_state == FileSegment::State::PARTIALLY_DOWNLOADED if (current_state == FileSegment::State::PARTIALLY_DOWNLOADED
|| current_state == FileSegment::State::PARTIALLY_DOWNLOADED_NO_CONTINUATION) || current_state == FileSegment::State::PARTIALLY_DOWNLOADED_NO_CONTINUATION)
{ {
checkForPartialDownload(); new_buf = checkForPartialDownload();
} }
if (!new_buf && read_type == ReadType::REMOTE_FS_READ_AND_PUT_IN_CACHE)
{
/**
* ReadType::REMOTE_FS_READ_AND_PUT_IN_CACHE means that on previous getReadBufferForFileSegment() call
* current buffer successfully called file_segment->getOrSetDownloader() and became a downloader
* for this file segment. However, the downloader's term has a lifespan of 1 nextImpl() call,
* e.g. downloader reads buffer_size byte and calls completeBatchAndResetDownloader() and some other
* thread can become a downloader if it calls getOrSetDownloader() faster.
*
* So downloader is committed to download only buffer_size bytes and then is not a downloader anymore,
* because there is no guarantee on a higher level, that current buffer will not disappear without
* being destructed till the end of query or without finishing the read range, which he was supposed
* to read by marks range given to him. Therefore, each nextImpl() call, in case of
* READ_AND_PUT_IN_CACHE, starts with getOrSetDownloader().
*/
impl = getReadBufferForFileSegment(*current_file_segment_it);
}
LOG_TEST(log, "Non-first initialization");
} }
else else
{ {
impl = getReadBufferForFileSegment(*current_file_segment_it); impl = getReadBufferForFileSegment(*current_file_segment_it);
LOG_TEST(log, "First initialization");
} }
if (use_external_buffer) if (use_external_buffer)
@ -444,52 +470,66 @@ bool CachedReadBufferFromRemoteFS::nextImpl()
/// download from offset a'' < a', but return buffer from offset a'. /// download from offset a'' < a', but return buffer from offset a'.
LOG_TEST(log, "Bytes to predownload: {}, caller_id: {}", bytes_to_predownload, FileSegment::getCallerId()); LOG_TEST(log, "Bytes to predownload: {}, caller_id: {}", bytes_to_predownload, FileSegment::getCallerId());
while (true) // while (true)
{ // {
if (bytes_to_predownload // if (!bytes_to_predownload || impl->eof())
&& file_segment->downloadOffset() + 1 != file_offset_of_buffer_end // {
&& !impl->eof()) // if (file_segment->downloadOffset() + 1 != file_offset_of_buffer_end)
{ // throw Exception(
result = impl->hasPendingData(); // ErrorCodes::LOGICAL_ERROR,
size = impl->available(); // "Failed to predownload. Current file segment: {}, current download offset: {}, expected: {}, eof: {}",
// file_segment->range().toString(), file_segment->downloadOffset(), file_offset_of_buffer_end, impl->eof());
break; // result = impl->hasPendingData();
// size = impl->available();
// size_t remaining_size_to_read = std::min(current_read_range.right, read_until_position - 1) - file_offset_of_buffer_end + 1;
// remaining_size_to_read = std::min(impl->available(), remaining_size_to_read);
// if (read_type == ReadType::CACHED && std::next(current_file_segment_it) == file_segments_holder->file_segments.end())
// {
// LOG_TEST(log, "Resize. Offset: {}, remaining size: {}, file offset: {}, range: {}",
// offset(), remaining_size_to_read, file_offset_of_buffer_end, file_segment->range().toString());
// impl->buffer().resize(offset() + remaining_size_to_read);
// }
// break;
// }
// if (file_segment->reserve(impl->buffer().size()))
// {
// size_t size_to_cache = std::min(bytes_to_predownload, impl->buffer().size());
// LOG_TEST(log, "Left to predownload: {}, buffer size: {}", bytes_to_predownload, impl->buffer().size());
// file_segment->write(impl->buffer().begin(), size_to_cache);
// bytes_to_predownload -= size_to_cache;
// impl->position() += size_to_cache;
// }
// else
// {
// file_segment->complete(FileSegment::State::PARTIALLY_DOWNLOADED_NO_CONTINUATION);
// bytes_to_predownload = 0;
// read_type = ReadType::REMOTE_FS_READ_BYPASS_CACHE;
// impl = getRemoteFSReadBuffer(file_segment, read_type);
// impl->seek(file_offset_of_buffer_end, SEEK_SET);
// LOG_TEST(
// log, "Predownload failed because of space limit. Will read from remote filesystem starting from offset: {}",
// file_offset_of_buffer_end);
// break;
// }
// }
} }
if (file_segment->reserve(impl->buffer().size())) auto download_current_segment = read_type == ReadType::REMOTE_FS_READ_AND_PUT_IN_CACHE;
{ if (download_current_segment != file_segment->isDownloader())
size_t size_to_cache = std::min(bytes_to_predownload, impl->buffer().size()); throw Exception(
LOG_TEST(log, "Left to predownload: {}, buffer size: {}", bytes_to_predownload, impl->buffer().size()); ErrorCodes::LOGICAL_ERROR,
"Incorrect segment state. Having read type: {}, Caller id: {}, downloader id: {}, file segment state: {}",
file_segment->write(impl->buffer().begin(), size_to_cache); toString(read_type), file_segment->getCallerId(), file_segment->getDownloader(), file_segment->state());
bytes_to_predownload -= size_to_cache;
impl->position() += size_to_cache;
}
else
{
file_segment->complete(FileSegment::State::PARTIALLY_DOWNLOADED_NO_CONTINUATION);
bytes_to_predownload = 0;
read_type = ReadType::REMOTE_FS_READ_BYPASS_CACHE;
impl = getRemoteFSReadBuffer(file_segment, read_type);
impl->seek(file_offset_of_buffer_end, SEEK_SET);
LOG_TEST(
log, "Predownload failed because of space limit. Will read from remote filesystem starting from offset: {}",
file_offset_of_buffer_end);
break;
}
if (file_segment->downloadOffset() + 1 != file_offset_of_buffer_end
&& read_type == ReadType::REMOTE_FS_AND_PUT_IN_CACHE)
throw Exception(ErrorCodes::LOGICAL_ERROR, "Predownloading failed");
}
}
auto download_current_segment = read_type == ReadType::REMOTE_FS_AND_PUT_IN_CACHE;
assert(!download_current_segment || file_segment->isDownloader());
try try
{ {
@ -497,6 +537,14 @@ bool CachedReadBufferFromRemoteFS::nextImpl()
{ {
result = impl->next(); result = impl->next();
size = impl->buffer().size(); size = impl->buffer().size();
size_t remaining_size_to_read = std::min(current_read_range.right, read_until_position - 1) - file_offset_of_buffer_end + 1;
if (read_type == ReadType::CACHED && std::next(current_file_segment_it) == file_segments_holder->file_segments.end())
{
size = std::min(size, remaining_size_to_read);
impl->buffer().resize(size);
}
} }
} }
catch (...) catch (...)
@ -504,7 +552,7 @@ bool CachedReadBufferFromRemoteFS::nextImpl()
tryLogCurrentException(__PRETTY_FUNCTION__); tryLogCurrentException(__PRETTY_FUNCTION__);
if (download_current_segment) if (download_current_segment)
file_segment->complete(FileSegment::State::PARTIALLY_DOWNLOADED); file_segment->complete(FileSegment::State::PARTIALLY_DOWNLOADED, true);
/// Note: If exception happens in another place -- out of scope of this buffer, then /// Note: If exception happens in another place -- out of scope of this buffer, then
/// downloader's FileSegmentsHolder is responsible to call file_segment->complete(). /// downloader's FileSegmentsHolder is responsible to call file_segment->complete().
@ -515,6 +563,8 @@ bool CachedReadBufferFromRemoteFS::nextImpl()
if (result) if (result)
{ {
file_offset_of_buffer_end += size;
if (download_current_segment) if (download_current_segment)
{ {
if (file_segment->reserve(size)) if (file_segment->reserve(size))
@ -528,31 +578,22 @@ bool CachedReadBufferFromRemoteFS::nextImpl()
} }
} }
/// just implement setReadUntilPosition() for local filesysteam read buffer?
if (read_type == ReadType::CACHED && std::next(current_file_segment_it) == file_segments_holder->file_segments.end())
{
size_t remaining_size_to_read = std::min(current_read_range.right, read_until_position - 1) - file_offset_of_buffer_end + 1;
impl->buffer().resize(std::min(impl->buffer().size(), remaining_size_to_read));
}
file_offset_of_buffer_end += impl->buffer().size();
switch (read_type) switch (read_type)
{ {
case ReadType::CACHED: case ReadType::CACHED:
{ {
ProfileEvents::increment(ProfileEvents::RemoteFSCacheReadBytes, working_buffer.size()); ProfileEvents::increment(ProfileEvents::RemoteFSCacheReadBytes, size);
break; break;
} }
case ReadType::REMOTE_FS_READ_BYPASS_CACHE: case ReadType::REMOTE_FS_READ_BYPASS_CACHE:
{ {
ProfileEvents::increment(ProfileEvents::RemoteFSReadBytes, working_buffer.size()); ProfileEvents::increment(ProfileEvents::RemoteFSReadBytes, size);
break; break;
} }
case ReadType::REMOTE_FS_AND_PUT_IN_CACHE: case ReadType::REMOTE_FS_READ_AND_PUT_IN_CACHE:
{ {
ProfileEvents::increment(ProfileEvents::RemoteFSReadBytes, working_buffer.size()); ProfileEvents::increment(ProfileEvents::RemoteFSReadBytes, size);
ProfileEvents::increment(ProfileEvents::RemoteFSCacheDownloadBytes, working_buffer.size()); ProfileEvents::increment(ProfileEvents::RemoteFSCacheDownloadBytes, size);
break; break;
} }
} }

View File

@ -38,7 +38,7 @@ private:
{ {
CACHED, CACHED,
REMOTE_FS_READ_BYPASS_CACHE, REMOTE_FS_READ_BYPASS_CACHE,
REMOTE_FS_AND_PUT_IN_CACHE, REMOTE_FS_READ_AND_PUT_IN_CACHE,
}; };
SeekableReadBufferPtr getRemoteFSReadBuffer(FileSegmentPtr file_segment, ReadType read_type_); SeekableReadBufferPtr getRemoteFSReadBuffer(FileSegmentPtr file_segment, ReadType read_type_);
@ -80,8 +80,8 @@ private:
return "CACHED"; return "CACHED";
case ReadType::REMOTE_FS_READ_BYPASS_CACHE: case ReadType::REMOTE_FS_READ_BYPASS_CACHE:
return "REMOTE_FS_READ_BYPASS_CACHE"; return "REMOTE_FS_READ_BYPASS_CACHE";
case ReadType::REMOTE_FS_AND_PUT_IN_CACHE: case ReadType::REMOTE_FS_READ_AND_PUT_IN_CACHE:
return "REMOTE_FS_AND_PUT_IN_CACHE"; return "REMOTE_FS_READ_AND_PUT_IN_CACHE";
} }
} }
}; };

View File

@ -160,7 +160,10 @@ off_t ReadBufferFromS3::seek(off_t offset_, int whence)
return offset; return offset;
if (impl && restricted_seek) if (impl && restricted_seek)
throw Exception("Seek is allowed only before first read attempt from the buffer.", ErrorCodes::CANNOT_SEEK_THROUGH_FILE); throw Exception(
ErrorCodes::CANNOT_SEEK_THROUGH_FILE,
"Seek is allowed only before first read attempt from the buffer (current offset: {}, new offset: {})",
offset, offset_);
if (whence != SEEK_SET) if (whence != SEEK_SET)
throw Exception("Only SEEK_SET mode is allowed.", ErrorCodes::CANNOT_SEEK_THROUGH_FILE); throw Exception("Only SEEK_SET mode is allowed.", ErrorCodes::CANNOT_SEEK_THROUGH_FILE);