159 typedef BufferT Buffer;
160 typedef DynamicParser<CompactBinaryReader&> Parser;
163 BOND_STATIC_CONSTEXPR uint16_t magic = COMPACT_PROTOCOL;
164 BOND_STATIC_CONSTEXPR uint16_t version = v2;
168 uint16_t version_value = default_version<CompactBinaryReader>::value)
170 _version(version_value)
172 BOOST_ASSERT(protocol_has_multiple_versions<CompactBinaryReader>::value
173 ? _version <= CompactBinaryReader::version
174 : _version == default_version<CompactBinaryReader>::value);
183 : _input(that._input),
184 _version(that._version)
191 return _input == rhs._input;
196 typename boost::call_traits<Buffer>::const_reference
204 typename boost::call_traits<Buffer>::reference
213 uint16_t magic_value;
215 _input.Read(magic_value);
216 _input.Read(_version);
218 return magic_value == CompactBinaryReader::magic
219 && (protocol_has_multiple_versions<CompactBinaryReader>::value
220 ? _version <= CompactBinaryReader::version
221 : _version == default_version<CompactBinaryReader>::value);
226 void ReadStructBegin(
bool base =
false)
228 if (!base && v2 == _version)
236 void ReadStructEnd(
bool =
false)
240 void ReadFieldBegin(BondDataType& type, uint16_t&
id)
246 type =
static_cast<BondDataType
>(raw & 0x1f);
247 id =
static_cast<uint16_t
>(raw & (0x07 << 5));
249 if (
id == (0x07 << 5))
254 else if (
id == (0x06 << 5))
258 id =
static_cast<uint16_t
>(raw);
273 void ReadContainerBegin(uint32_t& size, BondDataType& type)
278 type =
static_cast<BondDataType
>(raw & 0x1f);
280 if (v2 == _version && (raw & (0x07 << 5)))
281 size = (raw >> 5) - 1;
288 void ReadContainerBegin(uint32_t& size, std::pair<BondDataType, BondDataType>& type)
293 type.first =
static_cast<BondDataType
>(raw);
296 type.second =
static_cast<BondDataType
>(raw);
303 void ReadContainerEnd()
308 template <
typename T>
309 typename boost::enable_if<std::is_floating_point<T> >::type
316 template <
typename T>
317 typename boost::enable_if<std::is_unsigned<T> >::type
320 ReadVariableUnsigned(_input, value);
324 template <
typename T>
325 typename boost::enable_if<is_signed_int<T> >::type
328 typename std::make_unsigned<T>::type unsigned_value;
330 ReadVariableUnsigned(_input, unsigned_value);
331 value = DecodeZigZag(unsigned_value);
336 template <
typename T>
337 typename boost::enable_if<std::is_enum<T> >::type
340 BOOST_STATIC_ASSERT(
sizeof(value) ==
sizeof(int32_t));
343 std::memcpy(&value, &raw,
sizeof(raw));
348 void Read(int8_t& value)
355 void Read(uint8_t& value)
362 void Read(
bool& value)
369 template <
typename T>
370 typename boost::enable_if<is_string_type<T> >::type
377 constexpr uint8_t charSize =
static_cast<uint8_t
>(
sizeof(
typename detail::string_char_int_type<T>::type));
378 uint32_t numStringBytes = detail::checked_multiply(length, charSize);
379 if (!_input.CanRead(numStringBytes))
381 OutOfBoundStringSizeException();
384 detail::ReadStringData(_input, value, length);
389 void Read(blob& value, uint32_t size)
391 _input.Read(value, size);
396 bool CanReadArray(uint32_t num_elems)
400 return _input.CanRead(num_elems);
404 typename boost::enable_if<typename std::is_floating_point<T>::value,
bool>::type
405 CanReadArray(uint32_t num_elems)
409 uint64_t num_bytes =
static_cast<uint64_t
>(num_elems) *
sizeof(T);
414 return (num_bytes >> 32 == 0) && _input.CanRead(
static_cast<uint32_t
>(num_bytes & 0xffffffff));
417 template <
typename T>
420 SkipType<get_type_id<T>::value>();
423 template <
typename T>
424 void Skip(
const bonded<T, CompactBinaryReader&>&)
426 SkipType<bond::BT_STRUCT>();
429 void Skip(BondDataType type)
435 using BT = BondDataType;
438 typename boost::enable_if_c<(T == BT_BOOL || T == BT_UINT8 || T == BT_INT8)>::type
439 SkipType(uint32_t size = 1)
441 _input.Skip(detail::checked_multiply(size,
sizeof(uint8_t)));
445 typename boost::enable_if_c<(T == BT_UINT16 || T == BT_UINT32 || T == BT_UINT64
446 || T == BT_INT16 || T == BT_INT32 || T == BT_INT64)>::type
454 typename boost::enable_if_c<(T == BT_FLOAT)>::type
455 SkipType(uint32_t size = 1)
457 _input.Skip(detail::checked_multiply(size,
sizeof(
float)));
461 typename boost::enable_if_c<(T == BT_DOUBLE)>::type
462 SkipType(uint32_t size = 1)
464 _input.Skip(detail::checked_multiply(size,
sizeof(
double)));
468 typename boost::enable_if_c<(T == BT_STRING)>::type
478 typename boost::enable_if_c<(T == BT_WSTRING)>::type
484 _input.Skip(detail::checked_multiply(length,
sizeof(uint16_t)));
488 typename boost::enable_if_c<(T == BT_SET || T == BT_LIST)>::type
491 BondDataType element_type;
494 bond::detail::RecursionGuard guard;
496 ReadContainerBegin(size, element_type);
497 SkipType(element_type, size);
502 typename boost::enable_if_c<(T == BT_MAP)>::type
505 std::pair<BondDataType, BondDataType> element_type;
508 bond::detail::RecursionGuard guard;
510 ReadContainerBegin(size, element_type);
511 for (int64_t i = 0; i < size; ++i)
513 SkipType(element_type.first);
514 SkipType(element_type.second);
521 BOOST_ASSERT(v1 == _version);
523 bond::detail::RecursionGuard guard;
530 BondDataType field_type;
532 for (ReadFieldBegin(field_type,
id);
533 field_type != bond::BT_STOP && field_type != bond::BT_STOP_BASE;
534 ReadFieldEnd(), ReadFieldBegin(field_type,
id))
536 SkipType(field_type);
541 if (field_type == bond::BT_STOP)
548 BOOST_ASSERT(v2 == _version);
556 typename boost::enable_if_c<(T == BT_STRUCT)>::type
570 typename boost::enable_if_c<(T == BT_STRUCT)>::type
571 SkipType(uint32_t size)
575 for (int64_t i = 0; i < size; ++i)
582 for (int64_t i = 0; i < size; ++i)
590 typename boost::enable_if_c<(T == BT_UINT16 || T == BT_UINT32 || T == BT_UINT64
591 || T == BT_INT16 || T == BT_INT32 || T == BT_INT64
592 || T == BT_STRING || T == BT_WSTRING
593 || T == BT_SET || T == BT_LIST || T == BT_MAP)>::type
594 SkipType(uint32_t size)
596 for (int64_t i = 0; i < size; ++i)
602 template <
typename... Args>
603 void SkipType(BondDataType type, Args&&... args)
610 SkipType<BT_BOOL>(std::forward<Args>(args)...);
619 SkipType<BT_UINT64>(std::forward<Args>(args)...);
623 SkipType<BT_FLOAT>(std::forward<Args>(args)...);
627 SkipType<BT_DOUBLE>(std::forward<Args>(args)...);
631 SkipType<BT_STRING>(std::forward<Args>(args)...);
635 SkipType<BT_WSTRING>(std::forward<Args>(args)...);
640 SkipType<BT_SET>(std::forward<Args>(args)...);
644 SkipType<BT_MAP>(std::forward<Args>(args)...);
648 SkipType<BT_STRUCT>(std::forward<Args>(args)...);
652 bond::UnknownDataTypeException();
660 template <
typename Input,
typename Output>
663 const CompactBinaryWriter<Output>&);
683class CompactBinaryWriter
688 Pass1(CompactBinaryWriter* writer)
697 CompactBinaryWriter* writer;
700 using Counter = CompactBinaryCounter::type;
703 typedef BufferT Buffer;
705 typedef CompactBinaryWriter<Counter> Pass0;
710 uint16_t version = default_version<Reader>::value)
715 BOOST_ASSERT(protocol_has_multiple_versions<Reader>::value
716 ? _version <= Reader::version
717 : _version == default_version<Reader>::value);
724 _version(pass1._version)
729 typename boost::call_traits<Buffer>::reference
738 return v2 == _version && !_it;
742 Pass1 WithPass0(Pass0& pass0)
744 _it = pass0._lengths.begin();
750 _output.Write(Reader::magic);
751 _output.Write(_version);
755 void WriteStructBegin(
const Metadata& ,
bool base)
759 LengthBegin(_output);
763 void WriteStructEnd(
bool base =
false)
767 _output.Write(
static_cast<uint8_t
>(BT_STOP_BASE));
771 _output.Write(
static_cast<uint8_t
>(BT_STOP));
777 template <
typename T>
778 void WriteField(uint16_t
id,
const bond::Metadata& ,
const T& value)
780 WriteFieldBegin(get_type_id<T>::value,
id);
786 void WriteFieldBegin(BondDataType type, uint16_t
id, const ::bond::Metadata& )
788 WriteFieldBegin(type,
id);
791 void WriteFieldBegin(BondDataType type, uint16_t
id)
793 BOOST_ASSERT((type & 0x1f) == type);
797 _output.Write(
static_cast<uint8_t
>(type | ((
id) << 5)));
801 _output.Write(
static_cast<uint8_t
>(type | (0x06 << 5)));
802 _output.Write(
static_cast<uint8_t
>(
id));
806 _output.Write(
static_cast<uint8_t
>(type | (0x07 << 5)));
816 void WriteContainerBegin(uint32_t size, BondDataType type)
818 BOOST_ASSERT((type & 0x1f) == type);
820 if (v2 == _version && size < 7)
822 Write(
static_cast<uint8_t
>(type | ((size + 1) << 5)));
826 Write(
static_cast<uint8_t
>(type));
832 void WriteContainerBegin(uint32_t size, std::pair<BondDataType, BondDataType> type)
834 Write(
static_cast<uint8_t
>(type.first));
835 Write(
static_cast<uint8_t
>(type.second));
836 WriteVariableUnsigned(_output, size);
840 void WriteContainerEnd()
844 template <
typename T>
845 typename boost::enable_if<std::is_floating_point<T> >::type
846 Write(
const T& value)
848 _output.Write(value);
852 template <
typename T>
853 typename boost::enable_if<std::is_unsigned<T> >::type
854 Write(
const T& value)
856 WriteVariableUnsigned(_output, value);
860 template <
typename T>
861 typename boost::enable_if<is_signed_int<T> >::type
862 Write(
const T& value)
864 WriteVariableUnsigned(_output, EncodeZigZag(value));
868 template <
typename T>
869 typename boost::enable_if<std::is_enum<T> >::type
870 Write(
const T& value)
872 BOOST_STATIC_ASSERT(
sizeof(value) ==
sizeof(int32_t));
873 Write(
static_cast<int32_t
>(value));
877 void Write(
const int8_t& value)
879 _output.Write(value);
883 void Write(
const uint8_t& value)
885 _output.Write(value);
889 void Write(
const bool& value)
891 _output.Write(value);
895 template <
typename T>
896 typename boost::enable_if<is_string_type<T> >::type
897 Write(
const T& value)
899 uint32_t length = string_length(value);
902 detail::WriteStringData(_output, value, length);
906 void Write(
const blob& value)
908 _output.Write(value);
912 template <
typename Buffer>
913 friend class CompactBinaryWriter;
915 void LengthBegin(Counter& counter)
917 _stack.push(_lengths.size());
918 _lengths.push(counter.GetCount());
921 void LengthEnd(Counter& counter)
923 uint32_t& length = _lengths[_stack.pop()];
925 length = counter.GetCount() - length;
926 counter.WriteVariableUnsigned(length);
946 detail::SimpleArray<uint32_t> _stack;
947 detail::SimpleArray<uint32_t> _lengths;
949 template <
typename Input,
typename Output>
951 bool is_protocol_version_same(
const CompactBinaryReader<Input>&,
952 const CompactBinaryWriter<Output>&);