mirror of
https://github.com/ClickHouse/ClickHouse.git
synced 2024-12-20 05:05:38 +00:00
131 lines
4.0 KiB
C++
131 lines
4.0 KiB
C++
#include <IO/WriteHelpers.h>
|
|
#include <cinttypes>
|
|
#include <utility>
|
|
#include <Common/formatIPv6.h>
|
|
#include <base/hex.h>
|
|
|
|
|
|
namespace DB
|
|
{
|
|
|
|
template <typename IteratorSrc, typename IteratorDst>
|
|
void formatHex(IteratorSrc src, IteratorDst dst, size_t num_bytes)
|
|
{
|
|
size_t src_pos = 0;
|
|
size_t dst_pos = 0;
|
|
for (; src_pos < num_bytes; ++src_pos)
|
|
{
|
|
writeHexByteLowercase(src[src_pos], &dst[dst_pos]);
|
|
dst_pos += 2;
|
|
}
|
|
}
|
|
|
|
std::array<char, 36> formatUUID(const UUID & uuid)
|
|
{
|
|
std::array<char, 36> dst;
|
|
auto * dst_ptr = dst.data();
|
|
|
|
#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
|
|
const auto * src_ptr = reinterpret_cast<const UInt8 *>(&uuid);
|
|
const std::reverse_iterator src(src_ptr + 16);
|
|
#else
|
|
const auto * src = reinterpret_cast<const UInt8 *>(&uuid);
|
|
#endif
|
|
formatHex(src + 8, dst_ptr, 4);
|
|
dst[8] = '-';
|
|
formatHex(src + 12, dst_ptr + 9, 2);
|
|
dst[13] = '-';
|
|
formatHex(src + 14, dst_ptr + 14, 2);
|
|
dst[18] = '-';
|
|
formatHex(src, dst_ptr + 19, 2);
|
|
dst[23] = '-';
|
|
formatHex(src + 2, dst_ptr + 24, 6);
|
|
|
|
return dst;
|
|
}
|
|
|
|
void writeIPv4Text(const IPv4 & ip, WriteBuffer & buf)
|
|
{
|
|
size_t idx = (ip >> 24);
|
|
buf.write(one_byte_to_string_lookup_table[idx].first, one_byte_to_string_lookup_table[idx].second);
|
|
buf.write('.');
|
|
idx = (ip >> 16) & 0xFF;
|
|
buf.write(one_byte_to_string_lookup_table[idx].first, one_byte_to_string_lookup_table[idx].second);
|
|
buf.write('.');
|
|
idx = (ip >> 8) & 0xFF;
|
|
buf.write(one_byte_to_string_lookup_table[idx].first, one_byte_to_string_lookup_table[idx].second);
|
|
buf.write('.');
|
|
idx = ip & 0xFF;
|
|
buf.write(one_byte_to_string_lookup_table[idx].first, one_byte_to_string_lookup_table[idx].second);
|
|
}
|
|
|
|
void writeIPv6Text(const IPv6 & ip, WriteBuffer & buf)
|
|
{
|
|
char addr[IPV6_MAX_TEXT_LENGTH + 1] {};
|
|
char * paddr = addr;
|
|
|
|
formatIPv6(reinterpret_cast<const unsigned char *>(&ip), paddr);
|
|
buf.write(addr, paddr - addr - 1);
|
|
}
|
|
|
|
void writeException(const Exception & e, WriteBuffer & buf, bool with_stack_trace)
|
|
{
|
|
writeBinaryLittleEndian(e.code(), buf);
|
|
writeBinary(String(e.name()), buf);
|
|
writeBinary(e.displayText() + getExtraExceptionInfo(e), buf);
|
|
|
|
if (with_stack_trace)
|
|
writeBinary(e.getStackTraceString(), buf);
|
|
else
|
|
writeBinary(String(), buf);
|
|
|
|
bool has_nested = false;
|
|
writeBinary(has_nested, buf);
|
|
}
|
|
|
|
|
|
/// The same, but quotes apply only if there are characters that do not match the identifier without quotes
|
|
template <typename F>
|
|
static inline void writeProbablyQuotedStringImpl(StringRef s, WriteBuffer & buf, F && write_quoted_string)
|
|
{
|
|
if (isValidIdentifier(s.toView())
|
|
/// This are valid identifiers but are problematic if present unquoted in SQL query.
|
|
&& !(s.size == strlen("distinct") && 0 == strncasecmp(s.data, "distinct", strlen("distinct")))
|
|
&& !(s.size == strlen("all") && 0 == strncasecmp(s.data, "all", strlen("all")))
|
|
&& !(s.size == strlen("table") && 0 == strncasecmp(s.data, "table", strlen("table"))))
|
|
{
|
|
writeString(s, buf);
|
|
}
|
|
else
|
|
write_quoted_string(s, buf);
|
|
}
|
|
|
|
void writeProbablyBackQuotedString(StringRef s, WriteBuffer & buf)
|
|
{
|
|
writeProbablyQuotedStringImpl(s, buf, [](StringRef s_, WriteBuffer & buf_) { writeBackQuotedString(s_, buf_); });
|
|
}
|
|
|
|
void writeProbablyDoubleQuotedString(StringRef s, WriteBuffer & buf)
|
|
{
|
|
writeProbablyQuotedStringImpl(s, buf, [](StringRef s_, WriteBuffer & buf_) { writeDoubleQuotedString(s_, buf_); });
|
|
}
|
|
|
|
void writeProbablyBackQuotedStringMySQL(StringRef s, WriteBuffer & buf)
|
|
{
|
|
writeProbablyQuotedStringImpl(s, buf, [](StringRef s_, WriteBuffer & buf_) { writeBackQuotedStringMySQL(s_, buf_); });
|
|
}
|
|
|
|
void writePointerHex(const void * ptr, WriteBuffer & buf)
|
|
{
|
|
writeString("0x", buf);
|
|
char hex_str[2 * sizeof(ptr)];
|
|
writeHexUIntLowercase(reinterpret_cast<uintptr_t>(ptr), hex_str);
|
|
buf.write(hex_str, 2 * sizeof(ptr));
|
|
}
|
|
|
|
String fourSpaceIndent(size_t indent)
|
|
{
|
|
return std::string(indent * 4, ' ');
|
|
}
|
|
}
|