#pragma once #include #include #include #include #include #include #include #include #include #include #include #include #define WRITE_HELPERS_DEFAULT_FLOAT_PRECISION 6U /// 20 цифр или 19 цифр и знак #define WRITE_HELPERS_MAX_INT_WIDTH 20U namespace DB { /// Функции-помошники для форматированной записи inline void writeChar(char x, WriteBuffer & buf) { buf.nextIfAtEnd(); *buf.position() = x; ++buf.position(); } /// Запись POD-типа в native формате template inline void writePODBinary(const T & x, WriteBuffer & buf) { buf.write(reinterpret_cast(&x), sizeof(x)); } template inline void writeIntBinary(const T & x, WriteBuffer & buf) { writePODBinary(x, buf); } template inline void writeFloatBinary(const T & x, WriteBuffer & buf) { writePODBinary(x, buf); } inline void writeStringBinary(const std::string & s, DB::WriteBuffer & buf) { writeVarUInt(s.size(), buf); buf.write(s.data(), s.size()); } inline void writeBoolText(bool x, WriteBuffer & buf) { writeChar(x ? '1' : '0', buf); } template void writeIntText(T x, WriteBuffer & buf) { char tmp[WRITE_HELPERS_MAX_INT_WIDTH]; bool negative = false; if (x == 0) { writeChar('0', buf); return; } if (x < 0) { x = -x; negative = true; /// Особый случай для самого маленького отрицательного числа if (unlikely(x < 0)) { if (sizeof(x) == 1) buf.write("-128", 4); else if (sizeof(x) == 2) buf.write("-32768", 6); else if (sizeof(x) == 4) buf.write("-2147483648", 11); else buf.write("-9223372036854775808", 20); return; } } char * pos; for (pos = tmp + WRITE_HELPERS_MAX_INT_WIDTH - 1; x != 0; --pos) { *pos = '0' + x % 10; x /= 10; } if (negative) *pos = '-'; else ++pos; buf.write(pos, tmp + WRITE_HELPERS_MAX_INT_WIDTH - pos); } template void writeFloatText(T x, WriteBuffer & buf, unsigned precision = WRITE_HELPERS_DEFAULT_FLOAT_PRECISION) { char tmp[24]; int res = std::snprintf(tmp, 24, "%.*g", precision, x); if (res >= 24 || res <= 0) throw Exception("Cannot print float or double number", ErrorCodes::CANNOT_PRINT_FLOAT_OR_DOUBLE_NUMBER); buf.write(tmp, res); } inline void writeString(const String & s, WriteBuffer & buf) { buf.write(s.data(), s.size()); } template void writeAnyEscapedString(const char * begin, const char * end, WriteBuffer & buf) { for (const char * it = begin; it != end; ++it) { switch (*it) { case '\b': writeChar('\\', buf); writeChar('b', buf); break; case '\f': writeChar('\\', buf); writeChar('f', buf); break; case '\n': writeChar('\\', buf); writeChar('n', buf); break; case '\r': writeChar('\\', buf); writeChar('r', buf); break; case '\t': writeChar('\\', buf); writeChar('t', buf); break; case '\0': writeChar('\\', buf); writeChar('0', buf); break; case '\\': writeChar('\\', buf); writeChar('\\', buf); break; case c: writeChar('\\', buf); writeChar(c, buf); break; default: writeChar(*it, buf); } } } template void writeAnyEscapedString(const String & s, WriteBuffer & buf) { writeAnyEscapedString(s.data(), s.data() + s.size(), buf); } inline void writeEscapedString(const String & s, WriteBuffer & buf) { /// strpbrk в libc под Linux на процессорах с SSE 4.2 хорошо оптимизирована (этот if ускоряет код в 1.5 раза) if (NULL == strpbrk(s.data(), "\b\f\n\r\t\'\\") && strlen(s.data()) == s.size()) writeString(s, buf); else writeAnyEscapedString<'\''>(s, buf); } template void writeAnyQuotedString(const String & s, WriteBuffer & buf) { writeChar(c, buf); writeAnyEscapedString(s, buf); writeChar(c, buf); } inline void writeQuotedString(const String & s, WriteBuffer & buf) { writeAnyQuotedString<'\''>(s, buf); } /// Совместимо с JSON. inline void writeDoubleQuotedString(const String & s, WriteBuffer & buf) { writeAnyQuotedString<'"'>(s, buf); } /// Выводит строку в обратных кавычках, как идентификатор в MySQL. inline void writeBackQuotedString(const String & s, WriteBuffer & buf) { writeAnyQuotedString<'`'>(s, buf); } /// То же самое, но обратные кавычки применяются только при наличии символов, не подходящих для идентификатора без обратных кавычек. inline void writeProbablyBackQuotedString(const String & s, WriteBuffer & buf) { if (s.empty() || !((s[0] >= 'a' && s[0] <= 'z') || (s[0] >= 'A' && s[0] <= 'Z') || s[0] == '_')) writeBackQuotedString(s, buf); else { const char * pos = s.data() + 1; const char * end = s.data() + s.size(); for (; pos < end; ++pos) if (!((*pos >= 'a' && *pos <= 'z') || (*pos >= 'A' && *pos <= 'Z') || (*pos >= '0' && *pos <= '9') || *pos == '_')) break; if (pos != end) writeBackQuotedString(s, buf); else writeString(s, buf); } } /// в формате YYYY-MM-DD inline void writeDateText(Yandex::DayNum_t date, WriteBuffer & buf) { char s[10] = {'0', '0', '0', '0', '-', '0', '0', '-', '0', '0'}; if (unlikely(date > DATE_LUT_MAX_DAY_NUM || date == 0)) { buf.write(s, 10); return; } const Yandex::DateLUT::Values & values = Yandex::DateLUTSingleton::instance().getValues(date); s[0] += values.year / 1000; s[1] += (values.year / 100) % 10; s[2] += (values.year / 10) % 10; s[3] += values.year % 10; s[5] += values.month / 10; s[6] += values.month % 10; s[8] += values.day_of_month / 10; s[9] += values.day_of_month % 10; buf.write(s, 10); } inline void writeDateText(mysqlxx::Date date, WriteBuffer & buf) { char s[10] = {'0', '0', '0', '0', '-', '0', '0', '-', '0', '0'}; s[0] += date.year() / 1000; s[1] += (date.year() / 100) % 10; s[2] += (date.year() / 10) % 10; s[3] += date.year() % 10; s[5] += date.month() / 10; s[6] += date.month() % 10; s[8] += date.day() / 10; s[9] += date.day() % 10; buf.write(s, 10); } /// в формате YYYY-MM-DD HH:MM:SS, согласно текущему часовому поясу inline void writeDateTimeText(time_t datetime, WriteBuffer & buf) { char s[19] = {'0', '0', '0', '0', '-', '0', '0', '-', '0', '0', ' ', '0', '0', ':', '0', '0', ':', '0', '0'}; if (unlikely(datetime > DATE_LUT_MAX || datetime == 0)) { buf.write(s, 19); return; } Yandex::DateLUTSingleton & date_lut = Yandex::DateLUTSingleton::instance(); const Yandex::DateLUT::Values & values = date_lut.getValues(datetime); s[0] += values.year / 1000; s[1] += (values.year / 100) % 10; s[2] += (values.year / 10) % 10; s[3] += values.year % 10; s[5] += values.month / 10; s[6] += values.month % 10; s[8] += values.day_of_month / 10; s[9] += values.day_of_month % 10; UInt8 hour = date_lut.toHourInaccurate(datetime); UInt8 minute = date_lut.toMinuteInaccurate(datetime); UInt8 second = date_lut.toSecondInaccurate(datetime); s[11] += hour / 10; s[12] += hour % 10; s[14] += minute / 10; s[15] += minute % 10; s[17] += second / 10; s[18] += second % 10; buf.write(s, 19); } inline void writeDateTimeText(mysqlxx::DateTime datetime, WriteBuffer & buf) { char s[19] = {'0', '0', '0', '0', '-', '0', '0', '-', '0', '0', ' ', '0', '0', ':', '0', '0', ':', '0', '0'}; s[0] += datetime.year() / 1000; s[1] += (datetime.year() / 100) % 10; s[2] += (datetime.year() / 10) % 10; s[3] += datetime.year() % 10; s[5] += datetime.month() / 10; s[6] += datetime.month() % 10; s[8] += datetime.day() / 10; s[9] += datetime.day() % 10; s[11] += datetime.hour() / 10; s[12] += datetime.hour() % 10; s[14] += datetime.minute() / 10; s[15] += datetime.minute() % 10; s[17] += datetime.second() / 10; s[18] += datetime.second() % 10; buf.write(s, 19); } /// Вывести mysqlxx::Row в tab-separated виде inline void writeEscapedRow(const mysqlxx::Row & row, WriteBuffer & buf) { for (size_t i = 0; i < row.size(); ++i) { if (i != 0) buf.write('\t'); if (unlikely(row[i].isNull())) { buf.write("\\N", 2); continue; } writeAnyEscapedString<'\''>(row[i].data(), row[i].data() + row[i].length(), buf); } } /// Методы вывода в бинарном виде inline void writeBinary(const UInt8 & x, WriteBuffer & buf) { writePODBinary(x, buf); } inline void writeBinary(const UInt16 & x, WriteBuffer & buf) { writePODBinary(x, buf); } inline void writeBinary(const UInt32 & x, WriteBuffer & buf) { writePODBinary(x, buf); } inline void writeBinary(const UInt64 & x, WriteBuffer & buf) { writePODBinary(x, buf); } inline void writeBinary(const Int8 & x, WriteBuffer & buf) { writePODBinary(x, buf); } inline void writeBinary(const Int16 & x, WriteBuffer & buf) { writePODBinary(x, buf); } inline void writeBinary(const Int32 & x, WriteBuffer & buf) { writePODBinary(x, buf); } inline void writeBinary(const Int64 & x, WriteBuffer & buf) { writePODBinary(x, buf); } inline void writeBinary(const Float32 & x, WriteBuffer & buf) { writePODBinary(x, buf); } inline void writeBinary(const Float64 & x, WriteBuffer & buf) { writePODBinary(x, buf); } inline void writeBinary(const String & x, WriteBuffer & buf) { writeStringBinary(x, buf); } inline void writeBinary(const bool & x, WriteBuffer & buf) { writePODBinary(x, buf); } inline void writeBinary(const Yandex::VisitID_t & x, WriteBuffer & buf) { writePODBinary(static_cast(x), buf); } inline void writeBinary(const mysqlxx::Date & x, WriteBuffer & buf) { writePODBinary(x, buf); } inline void writeBinary(const mysqlxx::DateTime & x, WriteBuffer & buf) { writePODBinary(x, buf); } /// Методы для вывода значения в текстовом виде для tab-separated формата. inline void writeText(const UInt8 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeText(const UInt16 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeText(const UInt32 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeText(const UInt64 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeText(const Int8 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeText(const Int16 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeText(const Int32 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeText(const Int64 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeText(const Float32 & x, WriteBuffer & buf) { writeFloatText(x, buf); } inline void writeText(const Float64 & x, WriteBuffer & buf) { writeFloatText(x, buf); } inline void writeText(const String & x, WriteBuffer & buf) { writeEscapedString(x, buf); } inline void writeText(const bool & x, WriteBuffer & buf) { writeBoolText(x, buf); } inline void writeText(const Yandex::VisitID_t & x, WriteBuffer & buf) { writeIntText(static_cast(x), buf); } inline void writeText(const mysqlxx::Date & x, WriteBuffer & buf) { writeDateText(x, buf); } inline void writeText(const mysqlxx::DateTime & x, WriteBuffer & buf) { writeDateTimeText(x, buf); } /// Методы для вывода в текстовом виде в кавычках inline void writeQuoted(const UInt8 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeQuoted(const UInt16 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeQuoted(const UInt32 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeQuoted(const UInt64 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeQuoted(const Int8 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeQuoted(const Int16 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeQuoted(const Int32 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeQuoted(const Int64 & x, WriteBuffer & buf) { writeIntText(x, buf); } inline void writeQuoted(const Float32 & x, WriteBuffer & buf) { writeFloatText(x, buf); } inline void writeQuoted(const Float64 & x, WriteBuffer & buf) { writeFloatText(x, buf); } inline void writeQuoted(const String & x, WriteBuffer & buf) { writeQuotedString(x, buf); } inline void writeQuoted(const bool & x, WriteBuffer & buf) { writeBoolText(x, buf); } inline void writeQuoted(const Yandex::VisitID_t & x, WriteBuffer & buf) { writeIntText(static_cast(x), buf); } inline void writeQuoted(const mysqlxx::Date & x, WriteBuffer & buf) { writeChar('\'', buf); writeDateText(x, buf); writeChar('\'', buf); } inline void writeQuoted(const mysqlxx::DateTime & x, WriteBuffer & buf) { writeChar('\'', buf); writeDateTimeText(x, buf); writeChar('\'', buf); } /// Сериализация эксепшена (чтобы его можно было передать по сети) void writeException(const Exception & e, WriteBuffer & buf); }