#pragma once #include "chunked_encoder.hpp" #include "socket_stream.hpp" #include "bolt/v1/packing/codes.hpp" #include "bolt/v1/messaging/codes.hpp" #include "utils/types/byte.hpp" #include "utils/bswap.hpp" namespace bolt { template class BoltEncoder { static constexpr int64_t plus_2_to_the_31 = 2147483648L; static constexpr int64_t plus_2_to_the_15 = 32768L; static constexpr int64_t plus_2_to_the_7 = 128L; static constexpr int64_t minus_2_to_the_4 = -16L; static constexpr int64_t minus_2_to_the_7 = -128L; static constexpr int64_t minus_2_to_the_15 = -32768L; static constexpr int64_t minus_2_to_the_31 = -2147483648L; public: BoltEncoder(Socket& socket) : stream(socket) {} void flush() { encoder.flush(); } void write(byte value) { encoder.write(value); } void write(const byte* values, size_t n) { encoder.write(values, n); } void write_null() { encoder.write(pack::Null); } void write(bool value) { if(value) write_true(); else write_false(); } void write_true() { encoder.write(pack::True); } void write_false() { encoder.write(pack::False); } template void write_value(T value) { value = bswap(value); encoder.write(reinterpret_cast(&value), sizeof(value)); } void write_integer(int64_t value) { if(value >= minus_2_to_the_4 && value < plus_2_to_the_7) { write(static_cast(value)); } else if(value >= minus_2_to_the_7 && value < minus_2_to_the_4) { write(pack::Int8); write(static_cast(value)); } else if(value >= minus_2_to_the_15 && value < plus_2_to_the_15) { write(pack::Int16); write_value(static_cast(value)); } else if(value >= minus_2_to_the_31 && value < plus_2_to_the_31) { write(pack::Int32); write_value(static_cast(value)); } else { write(pack::Int64); write_value(value); } } void write(double value) { write(pack::Float64); write_value(*reinterpret_cast(&value)); } void write_map_header(size_t size) { if(size < 0x10) { write(static_cast(pack::TinyMap | size)); } else if(size <= 0xFF) { write(pack::Map8); write(static_cast(size)); } else if(size <= 0xFFFF) { write(pack::Map16); write_value(size); } else { write(pack::Map32); write_value(size); } } void write_empty_map() { write(pack::TinyMap); } void write_list_header(size_t size) { if(size < 0x10) { write(static_cast(pack::TinyList | size)); } else if(size <= 0xFF) { write(pack::List8); write(static_cast(size)); } else if(size <= 0xFFFF) { write(pack::List16); write_value(size); } else { write(pack::List32); write_value(size); } } void write_empty_list() { write(pack::TinyList); } void write_string_header(size_t size) { if(size < 0x10) { write(static_cast(pack::TinyString | size)); } else if(size <= 0xFF) { write(pack::String8); write(static_cast(size)); } else if(size <= 0xFFFF) { write(pack::String16); write_value(size); } else { write(pack::String32); write_value(size); } } void write_string(const std::string& str) { write_string(str.c_str(), str.size()); } void write_string(const char* str, size_t len) { write_string_header(len); write(reinterpret_cast(str), len); } void write_struct_header(size_t size) { if(size < 0x10) { write(static_cast(pack::TinyStruct | size)); } else if(size <= 0xFF) { write(pack::Struct8); write(static_cast(size)); } else { write(pack::Struct16); write_value(size); } } void message_success() { write_struct_header(1); write(underlying_cast(MessageCode::Success)); } void message_success_empty() { message_success(); write_empty_map(); } void message_record() { write_struct_header(1); write(underlying_cast(MessageCode::Record)); } void message_record_empty() { message_record(); write_empty_list(); } private: SocketStream stream; ChunkedEncoder encoder {stream}; }; }