2021-12-09 10:39:28 +00:00
|
|
|
#include <Storages/MergeTree/RequestResponse.h>
|
|
|
|
|
|
|
|
#include <Core/ProtocolDefines.h>
|
|
|
|
#include <Common/SipHash.h>
|
|
|
|
#include <IO/WriteHelpers.h>
|
|
|
|
#include <IO/ReadHelpers.h>
|
2022-11-14 06:13:42 +00:00
|
|
|
#include <IO/Operators.h>
|
2021-12-09 10:39:28 +00:00
|
|
|
|
|
|
|
#include <consistent_hashing.h>
|
|
|
|
|
2022-11-14 05:09:03 +00:00
|
|
|
|
2021-12-09 10:39:28 +00:00
|
|
|
namespace DB
|
|
|
|
{
|
|
|
|
|
|
|
|
namespace ErrorCodes
|
|
|
|
{
|
|
|
|
extern const int UNKNOWN_PROTOCOL;
|
2022-11-14 05:09:03 +00:00
|
|
|
extern const int BAD_ARGUMENTS;
|
2021-12-09 10:39:28 +00:00
|
|
|
}
|
|
|
|
|
2022-11-14 05:09:03 +00:00
|
|
|
static void readMarkRangesBinary(MarkRanges & ranges, ReadBuffer & buf)
|
2021-12-09 10:39:28 +00:00
|
|
|
{
|
|
|
|
size_t size = 0;
|
|
|
|
readVarUInt(size, buf);
|
|
|
|
|
2022-11-14 05:09:03 +00:00
|
|
|
if (size > DEFAULT_MAX_STRING_SIZE)
|
|
|
|
throw Exception(ErrorCodes::BAD_ARGUMENTS, "Too large ranges size: {}.", size);
|
2021-12-09 10:39:28 +00:00
|
|
|
|
|
|
|
ranges.resize(size);
|
|
|
|
for (size_t i = 0; i < size; ++i)
|
|
|
|
{
|
|
|
|
readBinary(ranges[i].begin, buf);
|
|
|
|
readBinary(ranges[i].end, buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void writeMarkRangesBinary(const MarkRanges & ranges, WriteBuffer & buf)
|
|
|
|
{
|
|
|
|
writeVarUInt(ranges.size(), buf);
|
|
|
|
|
|
|
|
for (const auto & [begin, end] : ranges)
|
|
|
|
{
|
|
|
|
writeBinary(begin, buf);
|
|
|
|
writeBinary(end, buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void PartitionReadRequest::serialize(WriteBuffer & out) const
|
|
|
|
{
|
|
|
|
/// Must be the first
|
|
|
|
writeVarUInt(DBMS_PARALLEL_REPLICAS_PROTOCOL_VERSION, out);
|
|
|
|
|
|
|
|
writeStringBinary(partition_id, out);
|
|
|
|
writeStringBinary(part_name, out);
|
|
|
|
writeStringBinary(projection_name, out);
|
|
|
|
|
|
|
|
writeVarInt(block_range.begin, out);
|
|
|
|
writeVarInt(block_range.end, out);
|
|
|
|
|
|
|
|
writeMarkRangesBinary(mark_ranges, out);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-11-14 06:13:42 +00:00
|
|
|
String PartitionReadRequest::toString() const
|
|
|
|
{
|
|
|
|
WriteBufferFromOwnString out;
|
|
|
|
out << "partition: " << partition_id << ", part: " << part_name;
|
|
|
|
if (!projection_name.empty())
|
|
|
|
out << ", projection: " << projection_name;
|
|
|
|
out << ", block range: [" << block_range.begin << ", " << block_range.end << "]";
|
|
|
|
out << ", mark ranges: ";
|
|
|
|
|
|
|
|
bool is_first = true;
|
|
|
|
for (const auto & [begin, end] : mark_ranges)
|
|
|
|
{
|
|
|
|
if (!is_first)
|
|
|
|
out << ", ";
|
|
|
|
out << "[" << begin << ", " << end << ")";
|
|
|
|
is_first = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return out.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-12-09 10:39:28 +00:00
|
|
|
void PartitionReadRequest::deserialize(ReadBuffer & in)
|
|
|
|
{
|
|
|
|
UInt64 version;
|
|
|
|
readVarUInt(version, in);
|
|
|
|
if (version != DBMS_PARALLEL_REPLICAS_PROTOCOL_VERSION)
|
|
|
|
throw Exception(ErrorCodes::UNKNOWN_PROTOCOL, "Protocol versions for parallel reading \
|
|
|
|
from replicas differ. Got: {}, supported version: {}",
|
|
|
|
version, DBMS_PARALLEL_REPLICAS_PROTOCOL_VERSION);
|
|
|
|
|
|
|
|
readStringBinary(partition_id, in);
|
|
|
|
readStringBinary(part_name, in);
|
|
|
|
readStringBinary(projection_name, in);
|
|
|
|
|
|
|
|
readVarInt(block_range.begin, in);
|
|
|
|
readVarInt(block_range.end, in);
|
|
|
|
|
|
|
|
readMarkRangesBinary(mark_ranges, in);
|
|
|
|
}
|
|
|
|
|
|
|
|
UInt64 PartitionReadRequest::getConsistentHash(size_t buckets) const
|
|
|
|
{
|
2022-11-14 05:09:03 +00:00
|
|
|
SipHash hash;
|
|
|
|
|
|
|
|
hash.update(partition_id.size());
|
2021-12-09 10:39:28 +00:00
|
|
|
hash.update(partition_id);
|
2022-11-14 05:09:03 +00:00
|
|
|
|
|
|
|
hash.update(part_name.size());
|
2021-12-09 10:39:28 +00:00
|
|
|
hash.update(part_name);
|
2022-11-14 05:09:03 +00:00
|
|
|
|
|
|
|
hash.update(projection_name.size());
|
2021-12-09 10:39:28 +00:00
|
|
|
hash.update(projection_name);
|
|
|
|
|
|
|
|
hash.update(block_range.begin);
|
|
|
|
hash.update(block_range.end);
|
|
|
|
|
2022-11-14 05:09:03 +00:00
|
|
|
hash.update(mark_ranges.size());
|
2021-12-09 10:39:28 +00:00
|
|
|
for (const auto & range : mark_ranges)
|
|
|
|
{
|
|
|
|
hash.update(range.begin);
|
|
|
|
hash.update(range.end);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ConsistentHashing(hash.get64(), buckets);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void PartitionReadResponse::serialize(WriteBuffer & out) const
|
|
|
|
{
|
|
|
|
/// Must be the first
|
|
|
|
writeVarUInt(DBMS_PARALLEL_REPLICAS_PROTOCOL_VERSION, out);
|
|
|
|
|
2022-11-14 05:09:03 +00:00
|
|
|
writeBinary(denied, out);
|
2021-12-09 10:39:28 +00:00
|
|
|
writeMarkRangesBinary(mark_ranges, out);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void PartitionReadResponse::deserialize(ReadBuffer & in)
|
|
|
|
{
|
|
|
|
UInt64 version;
|
|
|
|
readVarUInt(version, in);
|
|
|
|
if (version != DBMS_PARALLEL_REPLICAS_PROTOCOL_VERSION)
|
|
|
|
throw Exception(ErrorCodes::UNKNOWN_PROTOCOL, "Protocol versions for parallel reading \
|
|
|
|
from replicas differ. Got: {}, supported version: {}",
|
|
|
|
version, DBMS_PARALLEL_REPLICAS_PROTOCOL_VERSION);
|
|
|
|
|
|
|
|
UInt64 value;
|
|
|
|
readVarUInt(value, in);
|
|
|
|
denied = static_cast<bool>(value);
|
|
|
|
readMarkRangesBinary(mark_ranges, in);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|