🔨 overwork binary subtypes
This commit is contained in:
parent
ab6e76dd05
commit
dead99eb0e
8 changed files with 195 additions and 117 deletions
|
@ -227,8 +227,10 @@ class binary_reader
|
||||||
return sax->parse_error(chars_read, last_token, parse_error::create(112, chars_read, exception_message(input_format_t::bson, "byte array length cannot be negative, is " + std::to_string(len), "binary")));
|
return sax->parse_error(chars_read, last_token, parse_error::create(112, chars_read, exception_message(input_format_t::bson, "byte array length cannot be negative, is " + std::to_string(len), "binary")));
|
||||||
}
|
}
|
||||||
|
|
||||||
result.m_has_subtype = true; // All BSON binary values have a subtype
|
// All BSON binary values have a subtype
|
||||||
get_number<std::uint8_t>(input_format_t::bson, result.m_subtype);
|
std::uint8_t subtype;
|
||||||
|
get_number<std::uint8_t>(input_format_t::bson, subtype);
|
||||||
|
result.set_subtype(subtype);
|
||||||
|
|
||||||
return get_binary(input_format_t::bson, len, result);
|
return get_binary(input_format_t::bson, len, result);
|
||||||
}
|
}
|
||||||
|
@ -901,25 +903,29 @@ class binary_reader
|
||||||
case 0x58: // Binary data (one-byte uint8_t for n follows)
|
case 0x58: // Binary data (one-byte uint8_t for n follows)
|
||||||
{
|
{
|
||||||
std::uint8_t len;
|
std::uint8_t len;
|
||||||
return get_number(input_format_t::cbor, len) and get_binary(input_format_t::cbor, len, result);
|
return get_number(input_format_t::cbor, len) and
|
||||||
|
get_binary(input_format_t::cbor, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0x59: // Binary data (two-byte uint16_t for n follow)
|
case 0x59: // Binary data (two-byte uint16_t for n follow)
|
||||||
{
|
{
|
||||||
std::uint16_t len;
|
std::uint16_t len;
|
||||||
return get_number(input_format_t::cbor, len) and get_binary(input_format_t::cbor, len, result);
|
return get_number(input_format_t::cbor, len) and
|
||||||
|
get_binary(input_format_t::cbor, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0x5A: // Binary data (four-byte uint32_t for n follow)
|
case 0x5A: // Binary data (four-byte uint32_t for n follow)
|
||||||
{
|
{
|
||||||
std::uint32_t len;
|
std::uint32_t len;
|
||||||
return get_number(input_format_t::cbor, len) and get_binary(input_format_t::cbor, len, result);
|
return get_number(input_format_t::cbor, len) and
|
||||||
|
get_binary(input_format_t::cbor, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0x5B: // Binary data (eight-byte uint64_t for n follow)
|
case 0x5B: // Binary data (eight-byte uint64_t for n follow)
|
||||||
{
|
{
|
||||||
std::uint64_t len;
|
std::uint64_t len;
|
||||||
return get_number(input_format_t::cbor, len) and get_binary(input_format_t::cbor, len, result);
|
return get_number(input_format_t::cbor, len) and
|
||||||
|
get_binary(input_format_t::cbor, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0x5F: // Binary data (indefinite length)
|
case 0x5F: // Binary data (indefinite length)
|
||||||
|
@ -1501,81 +1507,104 @@ class binary_reader
|
||||||
*/
|
*/
|
||||||
bool get_msgpack_binary(internal_binary_t& result)
|
bool get_msgpack_binary(internal_binary_t& result)
|
||||||
{
|
{
|
||||||
|
// helper function to set the subtype
|
||||||
|
auto assign_and_return_true = [&result](std::int8_t subtype)
|
||||||
|
{
|
||||||
|
result.set_subtype(static_cast<std::uint8_t>(subtype));
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
switch (current)
|
switch (current)
|
||||||
{
|
{
|
||||||
case 0xC4: // bin 8
|
case 0xC4: // bin 8
|
||||||
{
|
{
|
||||||
std::uint8_t len;
|
std::uint8_t len;
|
||||||
return get_number(input_format_t::msgpack, len) and get_binary(input_format_t::msgpack, len, result);
|
return get_number(input_format_t::msgpack, len) and
|
||||||
|
get_binary(input_format_t::msgpack, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xC5: // bin 16
|
case 0xC5: // bin 16
|
||||||
{
|
{
|
||||||
std::uint16_t len;
|
std::uint16_t len;
|
||||||
return get_number(input_format_t::msgpack, len) and get_binary(input_format_t::msgpack, len, result);
|
return get_number(input_format_t::msgpack, len) and
|
||||||
|
get_binary(input_format_t::msgpack, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xC6: // bin 32
|
case 0xC6: // bin 32
|
||||||
{
|
{
|
||||||
std::uint32_t len;
|
std::uint32_t len;
|
||||||
return get_number(input_format_t::msgpack, len) and get_binary(input_format_t::msgpack, len, result);
|
return get_number(input_format_t::msgpack, len) and
|
||||||
|
get_binary(input_format_t::msgpack, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xC7: // ext 8
|
case 0xC7: // ext 8
|
||||||
{
|
{
|
||||||
std::uint8_t len;
|
std::uint8_t len;
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, len) and
|
return get_number(input_format_t::msgpack, len) and
|
||||||
get_number(input_format_t::msgpack, result.m_subtype) and
|
get_number(input_format_t::msgpack, subtype) and
|
||||||
get_binary(input_format_t::msgpack, len, result);
|
get_binary(input_format_t::msgpack, len, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xC8: // ext 16
|
case 0xC8: // ext 16
|
||||||
{
|
{
|
||||||
std::uint16_t len;
|
std::uint16_t len;
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, len) and
|
return get_number(input_format_t::msgpack, len) and
|
||||||
get_number(input_format_t::msgpack, result.m_subtype) and
|
get_number(input_format_t::msgpack, subtype) and
|
||||||
get_binary(input_format_t::msgpack, len, result);
|
get_binary(input_format_t::msgpack, len, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xC9: // ext 32
|
case 0xC9: // ext 32
|
||||||
{
|
{
|
||||||
std::uint32_t len;
|
std::uint32_t len;
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, len) and
|
return get_number(input_format_t::msgpack, len) and
|
||||||
get_number(input_format_t::msgpack, result.m_subtype) and
|
get_number(input_format_t::msgpack, subtype) and
|
||||||
get_binary(input_format_t::msgpack, len, result);
|
get_binary(input_format_t::msgpack, len, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xD4: // fixext 1
|
case 0xD4: // fixext 1
|
||||||
{
|
{
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, result.m_subtype) and get_binary(input_format_t::msgpack, 1, result);
|
return get_number(input_format_t::msgpack, subtype) and
|
||||||
|
get_binary(input_format_t::msgpack, 1, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xD5: // fixext 2
|
case 0xD5: // fixext 2
|
||||||
{
|
{
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, result.m_subtype) and get_binary(input_format_t::msgpack, 2, result);
|
return get_number(input_format_t::msgpack, subtype) and
|
||||||
|
get_binary(input_format_t::msgpack, 2, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xD6: // fixext 4
|
case 0xD6: // fixext 4
|
||||||
{
|
{
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, result.m_subtype) and get_binary(input_format_t::msgpack, 4, result);
|
return get_number(input_format_t::msgpack, subtype) and
|
||||||
|
get_binary(input_format_t::msgpack, 4, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xD7: // fixext 8
|
case 0xD7: // fixext 8
|
||||||
{
|
{
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, result.m_subtype) and get_binary(input_format_t::msgpack, 8, result);
|
return get_number(input_format_t::msgpack, subtype) and
|
||||||
|
get_binary(input_format_t::msgpack, 8, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xD8: // fixext 16
|
case 0xD8: // fixext 16
|
||||||
{
|
{
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, result.m_subtype) and get_binary(input_format_t::msgpack, 16, result);
|
return get_number(input_format_t::msgpack, subtype) and
|
||||||
|
get_binary(input_format_t::msgpack, 16, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
default: // LCOV_EXCL_LINE
|
default: // LCOV_EXCL_LINE
|
||||||
|
|
|
@ -27,7 +27,7 @@ struct json_sax
|
||||||
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
||||||
using number_float_t = typename BasicJsonType::number_float_t;
|
using number_float_t = typename BasicJsonType::number_float_t;
|
||||||
using string_t = typename BasicJsonType::string_t;
|
using string_t = typename BasicJsonType::string_t;
|
||||||
using binary_t = typename BasicJsonType::binary_t;
|
using internal_binary_t = typename BasicJsonType::internal_binary_t;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
@brief a null value was read
|
@brief a null value was read
|
||||||
|
@ -78,7 +78,7 @@ struct json_sax
|
||||||
@return whether parsing should proceed
|
@return whether parsing should proceed
|
||||||
@note It is safe to move the passed binary.
|
@note It is safe to move the passed binary.
|
||||||
*/
|
*/
|
||||||
virtual bool binary(binary_t& val) = 0;
|
virtual bool binary(internal_binary_t& val) = 0;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
@brief the beginning of an object was read
|
@brief the beginning of an object was read
|
||||||
|
@ -154,7 +154,7 @@ class json_sax_dom_parser
|
||||||
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
||||||
using number_float_t = typename BasicJsonType::number_float_t;
|
using number_float_t = typename BasicJsonType::number_float_t;
|
||||||
using string_t = typename BasicJsonType::string_t;
|
using string_t = typename BasicJsonType::string_t;
|
||||||
using binary_t = typename BasicJsonType::binary_t;
|
using internal_binary_t = typename BasicJsonType::internal_binary_t;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
@param[in, out] r reference to a JSON value that is manipulated while
|
@param[in, out] r reference to a JSON value that is manipulated while
|
||||||
|
@ -208,9 +208,9 @@ class json_sax_dom_parser
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool binary(binary_t& val)
|
bool binary(internal_binary_t& val)
|
||||||
{
|
{
|
||||||
handle_value(BasicJsonType::binary_array(std::move(val)));
|
handle_value(std::move(val));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -343,7 +343,7 @@ class json_sax_dom_callback_parser
|
||||||
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
||||||
using number_float_t = typename BasicJsonType::number_float_t;
|
using number_float_t = typename BasicJsonType::number_float_t;
|
||||||
using string_t = typename BasicJsonType::string_t;
|
using string_t = typename BasicJsonType::string_t;
|
||||||
using binary_t = typename BasicJsonType::binary_t;
|
using internal_binary_t = typename BasicJsonType::internal_binary_t;
|
||||||
using parser_callback_t = typename BasicJsonType::parser_callback_t;
|
using parser_callback_t = typename BasicJsonType::parser_callback_t;
|
||||||
using parse_event_t = typename BasicJsonType::parse_event_t;
|
using parse_event_t = typename BasicJsonType::parse_event_t;
|
||||||
|
|
||||||
|
@ -398,9 +398,9 @@ class json_sax_dom_callback_parser
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool binary(binary_t& val)
|
bool binary(internal_binary_t& val)
|
||||||
{
|
{
|
||||||
handle_value(BasicJsonType::binary_array(val));
|
handle_value(std::move(val));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -654,7 +654,7 @@ class json_sax_acceptor
|
||||||
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
||||||
using number_float_t = typename BasicJsonType::number_float_t;
|
using number_float_t = typename BasicJsonType::number_float_t;
|
||||||
using string_t = typename BasicJsonType::string_t;
|
using string_t = typename BasicJsonType::string_t;
|
||||||
using binary_t = typename BasicJsonType::binary_t;
|
using internal_binary_t = typename BasicJsonType::internal_binary_t;
|
||||||
|
|
||||||
bool null()
|
bool null()
|
||||||
{
|
{
|
||||||
|
@ -686,7 +686,7 @@ class json_sax_acceptor
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool binary(binary_t& /*unused*/)
|
bool binary(internal_binary_t& /*unused*/)
|
||||||
{
|
{
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -658,7 +658,7 @@ class binary_writer
|
||||||
// step 1.5: if this is an ext type, write the subtype
|
// step 1.5: if this is an ext type, write the subtype
|
||||||
if (use_ext)
|
if (use_ext)
|
||||||
{
|
{
|
||||||
write_number(j.m_value.binary->subtype());
|
write_number(static_cast<std::int8_t>(j.m_value.binary->subtype()));
|
||||||
}
|
}
|
||||||
|
|
||||||
// step 2: write the byte string
|
// step 2: write the byte string
|
||||||
|
|
|
@ -12,42 +12,37 @@ namespace detail
|
||||||
@brief an internal type for a backed binary type
|
@brief an internal type for a backed binary type
|
||||||
|
|
||||||
This type is designed to be `binary_t` but with the subtype implementation
|
This type is designed to be `binary_t` but with the subtype implementation
|
||||||
detail. This type exists so that the user does not have to specify a struct
|
detail. This type exists so that the user does not have to specify a type
|
||||||
his- or herself with a specific naming scheme in order to override the
|
themselves with a specific naming scheme in order to override the binary type.
|
||||||
binary type. I.e. it's for ease of use.
|
|
||||||
*/
|
*/
|
||||||
template<typename BinaryType>
|
template<typename BinaryType>
|
||||||
class wrapped_binary_t : public BinaryType
|
class wrapped_binary_t : public BinaryType
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
// to simplify code in binary_reader
|
|
||||||
template<typename BasicJsonType, typename InputAdapterType, typename SAX>
|
|
||||||
friend class binary_reader;
|
|
||||||
|
|
||||||
using binary_t = BinaryType;
|
using binary_t = BinaryType;
|
||||||
|
|
||||||
wrapped_binary_t() noexcept(noexcept(binary_t()))
|
wrapped_binary_t() noexcept(noexcept(binary_t()))
|
||||||
: binary_t()
|
: binary_t()
|
||||||
{}
|
{}
|
||||||
|
|
||||||
wrapped_binary_t(const binary_t& bint) noexcept(noexcept(binary_t(bint)))
|
wrapped_binary_t(const binary_t& b) noexcept(noexcept(binary_t(b)))
|
||||||
: binary_t(bint)
|
: binary_t(b)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
wrapped_binary_t(binary_t&& bint) noexcept(noexcept(binary_t(std::move(bint))))
|
wrapped_binary_t(binary_t&& b) noexcept(noexcept(binary_t(std::move(b))))
|
||||||
: binary_t(std::move(bint))
|
: binary_t(std::move(b))
|
||||||
{}
|
{}
|
||||||
|
|
||||||
wrapped_binary_t(const binary_t& bint,
|
wrapped_binary_t(const binary_t& b,
|
||||||
std::uint8_t st) noexcept(noexcept(binary_t(bint)))
|
std::uint8_t subtype) noexcept(noexcept(binary_t(b)))
|
||||||
: binary_t(bint)
|
: binary_t(b)
|
||||||
, m_subtype(st)
|
, m_subtype(subtype)
|
||||||
, m_has_subtype(true)
|
, m_has_subtype(true)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
wrapped_binary_t(binary_t&& bint, std::uint8_t st) noexcept(noexcept(binary_t(std::move(bint))))
|
wrapped_binary_t(binary_t&& b, std::uint8_t subtype) noexcept(noexcept(binary_t(std::move(b))))
|
||||||
: binary_t(std::move(bint))
|
: binary_t(std::move(b))
|
||||||
, m_subtype(st)
|
, m_subtype(subtype)
|
||||||
, m_has_subtype(true)
|
, m_has_subtype(true)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
|
|
|
@ -900,6 +900,21 @@ class basic_json
|
||||||
While @ref binary_t is used to define how binary values are stored, this
|
While @ref binary_t is used to define how binary values are stored, this
|
||||||
type is used to access binary values once they are parsed.
|
type is used to access binary values once they are parsed.
|
||||||
|
|
||||||
|
Notes on subtypes:
|
||||||
|
|
||||||
|
- CBOR
|
||||||
|
- Binary values are represented as byte strings. No subtypes are
|
||||||
|
supported and will be ignored when CBOR is written.
|
||||||
|
- MessagePack
|
||||||
|
- If a subtype is given and the binary array contains exactly 1, 2, 4, 8,
|
||||||
|
or 16 elements, the fixext family (fixext1, fixext2, fixext4, fixext8)
|
||||||
|
is used. For other sizes, the ext family (ext8, ext16, ext32) is used.
|
||||||
|
The subtype is then added as singed 8-bit integer.
|
||||||
|
- If no subtype is given, the bin family (bin8, bin16, bin32) is used.
|
||||||
|
- BSON
|
||||||
|
- If a subtype is given, it is used and added as unsigned 8-bit integer.
|
||||||
|
- If no subtype is given, the generic binary subtype 0x00 is used.
|
||||||
|
|
||||||
@sa @ref binary_array -- create a binary array
|
@sa @ref binary_array -- create a binary array
|
||||||
*/
|
*/
|
||||||
using internal_binary_t = nlohmann::detail::wrapped_binary_t<BinaryType>;
|
using internal_binary_t = nlohmann::detail::wrapped_binary_t<BinaryType>;
|
||||||
|
|
|
@ -4764,7 +4764,7 @@ struct json_sax
|
||||||
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
||||||
using number_float_t = typename BasicJsonType::number_float_t;
|
using number_float_t = typename BasicJsonType::number_float_t;
|
||||||
using string_t = typename BasicJsonType::string_t;
|
using string_t = typename BasicJsonType::string_t;
|
||||||
using binary_t = typename BasicJsonType::binary_t;
|
using internal_binary_t = typename BasicJsonType::internal_binary_t;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
@brief a null value was read
|
@brief a null value was read
|
||||||
|
@ -4815,7 +4815,7 @@ struct json_sax
|
||||||
@return whether parsing should proceed
|
@return whether parsing should proceed
|
||||||
@note It is safe to move the passed binary.
|
@note It is safe to move the passed binary.
|
||||||
*/
|
*/
|
||||||
virtual bool binary(binary_t& val) = 0;
|
virtual bool binary(internal_binary_t& val) = 0;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
@brief the beginning of an object was read
|
@brief the beginning of an object was read
|
||||||
|
@ -4891,7 +4891,7 @@ class json_sax_dom_parser
|
||||||
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
||||||
using number_float_t = typename BasicJsonType::number_float_t;
|
using number_float_t = typename BasicJsonType::number_float_t;
|
||||||
using string_t = typename BasicJsonType::string_t;
|
using string_t = typename BasicJsonType::string_t;
|
||||||
using binary_t = typename BasicJsonType::binary_t;
|
using internal_binary_t = typename BasicJsonType::internal_binary_t;
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
@param[in, out] r reference to a JSON value that is manipulated while
|
@param[in, out] r reference to a JSON value that is manipulated while
|
||||||
|
@ -4945,9 +4945,9 @@ class json_sax_dom_parser
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool binary(binary_t& val)
|
bool binary(internal_binary_t& val)
|
||||||
{
|
{
|
||||||
handle_value(BasicJsonType::binary_array(std::move(val)));
|
handle_value(std::move(val));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5080,7 +5080,7 @@ class json_sax_dom_callback_parser
|
||||||
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
||||||
using number_float_t = typename BasicJsonType::number_float_t;
|
using number_float_t = typename BasicJsonType::number_float_t;
|
||||||
using string_t = typename BasicJsonType::string_t;
|
using string_t = typename BasicJsonType::string_t;
|
||||||
using binary_t = typename BasicJsonType::binary_t;
|
using internal_binary_t = typename BasicJsonType::internal_binary_t;
|
||||||
using parser_callback_t = typename BasicJsonType::parser_callback_t;
|
using parser_callback_t = typename BasicJsonType::parser_callback_t;
|
||||||
using parse_event_t = typename BasicJsonType::parse_event_t;
|
using parse_event_t = typename BasicJsonType::parse_event_t;
|
||||||
|
|
||||||
|
@ -5135,9 +5135,9 @@ class json_sax_dom_callback_parser
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool binary(binary_t& val)
|
bool binary(internal_binary_t& val)
|
||||||
{
|
{
|
||||||
handle_value(BasicJsonType::binary_array(val));
|
handle_value(std::move(val));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5391,7 +5391,7 @@ class json_sax_acceptor
|
||||||
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
using number_unsigned_t = typename BasicJsonType::number_unsigned_t;
|
||||||
using number_float_t = typename BasicJsonType::number_float_t;
|
using number_float_t = typename BasicJsonType::number_float_t;
|
||||||
using string_t = typename BasicJsonType::string_t;
|
using string_t = typename BasicJsonType::string_t;
|
||||||
using binary_t = typename BasicJsonType::binary_t;
|
using internal_binary_t = typename BasicJsonType::internal_binary_t;
|
||||||
|
|
||||||
bool null()
|
bool null()
|
||||||
{
|
{
|
||||||
|
@ -5423,7 +5423,7 @@ class json_sax_acceptor
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool binary(binary_t& /*unused*/)
|
bool binary(internal_binary_t& /*unused*/)
|
||||||
{
|
{
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -5820,8 +5820,10 @@ class binary_reader
|
||||||
return sax->parse_error(chars_read, last_token, parse_error::create(112, chars_read, exception_message(input_format_t::bson, "byte array length cannot be negative, is " + std::to_string(len), "binary")));
|
return sax->parse_error(chars_read, last_token, parse_error::create(112, chars_read, exception_message(input_format_t::bson, "byte array length cannot be negative, is " + std::to_string(len), "binary")));
|
||||||
}
|
}
|
||||||
|
|
||||||
result.m_has_subtype = true; // All BSON binary values have a subtype
|
// All BSON binary values have a subtype
|
||||||
get_number<std::uint8_t>(input_format_t::bson, result.m_subtype);
|
std::uint8_t subtype;
|
||||||
|
get_number<std::uint8_t>(input_format_t::bson, subtype);
|
||||||
|
result.set_subtype(subtype);
|
||||||
|
|
||||||
return get_binary(input_format_t::bson, len, result);
|
return get_binary(input_format_t::bson, len, result);
|
||||||
}
|
}
|
||||||
|
@ -6494,25 +6496,29 @@ class binary_reader
|
||||||
case 0x58: // Binary data (one-byte uint8_t for n follows)
|
case 0x58: // Binary data (one-byte uint8_t for n follows)
|
||||||
{
|
{
|
||||||
std::uint8_t len;
|
std::uint8_t len;
|
||||||
return get_number(input_format_t::cbor, len) and get_binary(input_format_t::cbor, len, result);
|
return get_number(input_format_t::cbor, len) and
|
||||||
|
get_binary(input_format_t::cbor, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0x59: // Binary data (two-byte uint16_t for n follow)
|
case 0x59: // Binary data (two-byte uint16_t for n follow)
|
||||||
{
|
{
|
||||||
std::uint16_t len;
|
std::uint16_t len;
|
||||||
return get_number(input_format_t::cbor, len) and get_binary(input_format_t::cbor, len, result);
|
return get_number(input_format_t::cbor, len) and
|
||||||
|
get_binary(input_format_t::cbor, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0x5A: // Binary data (four-byte uint32_t for n follow)
|
case 0x5A: // Binary data (four-byte uint32_t for n follow)
|
||||||
{
|
{
|
||||||
std::uint32_t len;
|
std::uint32_t len;
|
||||||
return get_number(input_format_t::cbor, len) and get_binary(input_format_t::cbor, len, result);
|
return get_number(input_format_t::cbor, len) and
|
||||||
|
get_binary(input_format_t::cbor, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0x5B: // Binary data (eight-byte uint64_t for n follow)
|
case 0x5B: // Binary data (eight-byte uint64_t for n follow)
|
||||||
{
|
{
|
||||||
std::uint64_t len;
|
std::uint64_t len;
|
||||||
return get_number(input_format_t::cbor, len) and get_binary(input_format_t::cbor, len, result);
|
return get_number(input_format_t::cbor, len) and
|
||||||
|
get_binary(input_format_t::cbor, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0x5F: // Binary data (indefinite length)
|
case 0x5F: // Binary data (indefinite length)
|
||||||
|
@ -7094,81 +7100,104 @@ class binary_reader
|
||||||
*/
|
*/
|
||||||
bool get_msgpack_binary(internal_binary_t& result)
|
bool get_msgpack_binary(internal_binary_t& result)
|
||||||
{
|
{
|
||||||
|
// helper function to set the subtype
|
||||||
|
auto assign_and_return_true = [&result](std::int8_t subtype)
|
||||||
|
{
|
||||||
|
result.set_subtype(static_cast<std::uint8_t>(subtype));
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
switch (current)
|
switch (current)
|
||||||
{
|
{
|
||||||
case 0xC4: // bin 8
|
case 0xC4: // bin 8
|
||||||
{
|
{
|
||||||
std::uint8_t len;
|
std::uint8_t len;
|
||||||
return get_number(input_format_t::msgpack, len) and get_binary(input_format_t::msgpack, len, result);
|
return get_number(input_format_t::msgpack, len) and
|
||||||
|
get_binary(input_format_t::msgpack, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xC5: // bin 16
|
case 0xC5: // bin 16
|
||||||
{
|
{
|
||||||
std::uint16_t len;
|
std::uint16_t len;
|
||||||
return get_number(input_format_t::msgpack, len) and get_binary(input_format_t::msgpack, len, result);
|
return get_number(input_format_t::msgpack, len) and
|
||||||
|
get_binary(input_format_t::msgpack, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xC6: // bin 32
|
case 0xC6: // bin 32
|
||||||
{
|
{
|
||||||
std::uint32_t len;
|
std::uint32_t len;
|
||||||
return get_number(input_format_t::msgpack, len) and get_binary(input_format_t::msgpack, len, result);
|
return get_number(input_format_t::msgpack, len) and
|
||||||
|
get_binary(input_format_t::msgpack, len, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xC7: // ext 8
|
case 0xC7: // ext 8
|
||||||
{
|
{
|
||||||
std::uint8_t len;
|
std::uint8_t len;
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, len) and
|
return get_number(input_format_t::msgpack, len) and
|
||||||
get_number(input_format_t::msgpack, result.m_subtype) and
|
get_number(input_format_t::msgpack, subtype) and
|
||||||
get_binary(input_format_t::msgpack, len, result);
|
get_binary(input_format_t::msgpack, len, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xC8: // ext 16
|
case 0xC8: // ext 16
|
||||||
{
|
{
|
||||||
std::uint16_t len;
|
std::uint16_t len;
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, len) and
|
return get_number(input_format_t::msgpack, len) and
|
||||||
get_number(input_format_t::msgpack, result.m_subtype) and
|
get_number(input_format_t::msgpack, subtype) and
|
||||||
get_binary(input_format_t::msgpack, len, result);
|
get_binary(input_format_t::msgpack, len, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xC9: // ext 32
|
case 0xC9: // ext 32
|
||||||
{
|
{
|
||||||
std::uint32_t len;
|
std::uint32_t len;
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, len) and
|
return get_number(input_format_t::msgpack, len) and
|
||||||
get_number(input_format_t::msgpack, result.m_subtype) and
|
get_number(input_format_t::msgpack, subtype) and
|
||||||
get_binary(input_format_t::msgpack, len, result);
|
get_binary(input_format_t::msgpack, len, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xD4: // fixext 1
|
case 0xD4: // fixext 1
|
||||||
{
|
{
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, result.m_subtype) and get_binary(input_format_t::msgpack, 1, result);
|
return get_number(input_format_t::msgpack, subtype) and
|
||||||
|
get_binary(input_format_t::msgpack, 1, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xD5: // fixext 2
|
case 0xD5: // fixext 2
|
||||||
{
|
{
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, result.m_subtype) and get_binary(input_format_t::msgpack, 2, result);
|
return get_number(input_format_t::msgpack, subtype) and
|
||||||
|
get_binary(input_format_t::msgpack, 2, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xD6: // fixext 4
|
case 0xD6: // fixext 4
|
||||||
{
|
{
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, result.m_subtype) and get_binary(input_format_t::msgpack, 4, result);
|
return get_number(input_format_t::msgpack, subtype) and
|
||||||
|
get_binary(input_format_t::msgpack, 4, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xD7: // fixext 8
|
case 0xD7: // fixext 8
|
||||||
{
|
{
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, result.m_subtype) and get_binary(input_format_t::msgpack, 8, result);
|
return get_number(input_format_t::msgpack, subtype) and
|
||||||
|
get_binary(input_format_t::msgpack, 8, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
case 0xD8: // fixext 16
|
case 0xD8: // fixext 16
|
||||||
{
|
{
|
||||||
result.m_has_subtype = true;
|
std::int8_t subtype;
|
||||||
return get_number(input_format_t::msgpack, result.m_subtype) and get_binary(input_format_t::msgpack, 16, result);
|
return get_number(input_format_t::msgpack, subtype) and
|
||||||
|
get_binary(input_format_t::msgpack, 16, result) and
|
||||||
|
assign_and_return_true(subtype);
|
||||||
}
|
}
|
||||||
|
|
||||||
default: // LCOV_EXCL_LINE
|
default: // LCOV_EXCL_LINE
|
||||||
|
@ -12682,7 +12711,7 @@ class binary_writer
|
||||||
// step 1.5: if this is an ext type, write the subtype
|
// step 1.5: if this is an ext type, write the subtype
|
||||||
if (use_ext)
|
if (use_ext)
|
||||||
{
|
{
|
||||||
write_number(j.m_value.binary->subtype());
|
write_number(static_cast<std::int8_t>(j.m_value.binary->subtype()));
|
||||||
}
|
}
|
||||||
|
|
||||||
// step 2: write the byte string
|
// step 2: write the byte string
|
||||||
|
@ -15631,42 +15660,37 @@ namespace detail
|
||||||
@brief an internal type for a backed binary type
|
@brief an internal type for a backed binary type
|
||||||
|
|
||||||
This type is designed to be `binary_t` but with the subtype implementation
|
This type is designed to be `binary_t` but with the subtype implementation
|
||||||
detail. This type exists so that the user does not have to specify a struct
|
detail. This type exists so that the user does not have to specify a type
|
||||||
his- or herself with a specific naming scheme in order to override the
|
themselves with a specific naming scheme in order to override the binary type.
|
||||||
binary type. I.e. it's for ease of use.
|
|
||||||
*/
|
*/
|
||||||
template<typename BinaryType>
|
template<typename BinaryType>
|
||||||
class wrapped_binary_t : public BinaryType
|
class wrapped_binary_t : public BinaryType
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
// to simplify code in binary_reader
|
|
||||||
template<typename BasicJsonType, typename InputAdapterType, typename SAX>
|
|
||||||
friend class binary_reader;
|
|
||||||
|
|
||||||
using binary_t = BinaryType;
|
using binary_t = BinaryType;
|
||||||
|
|
||||||
wrapped_binary_t() noexcept(noexcept(binary_t()))
|
wrapped_binary_t() noexcept(noexcept(binary_t()))
|
||||||
: binary_t()
|
: binary_t()
|
||||||
{}
|
{}
|
||||||
|
|
||||||
wrapped_binary_t(const binary_t& bint) noexcept(noexcept(binary_t(bint)))
|
wrapped_binary_t(const binary_t& b) noexcept(noexcept(binary_t(b)))
|
||||||
: binary_t(bint)
|
: binary_t(b)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
wrapped_binary_t(binary_t&& bint) noexcept(noexcept(binary_t(std::move(bint))))
|
wrapped_binary_t(binary_t&& b) noexcept(noexcept(binary_t(std::move(b))))
|
||||||
: binary_t(std::move(bint))
|
: binary_t(std::move(b))
|
||||||
{}
|
{}
|
||||||
|
|
||||||
wrapped_binary_t(const binary_t& bint,
|
wrapped_binary_t(const binary_t& b,
|
||||||
std::uint8_t st) noexcept(noexcept(binary_t(bint)))
|
std::uint8_t subtype) noexcept(noexcept(binary_t(b)))
|
||||||
: binary_t(bint)
|
: binary_t(b)
|
||||||
, m_subtype(st)
|
, m_subtype(subtype)
|
||||||
, m_has_subtype(true)
|
, m_has_subtype(true)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
wrapped_binary_t(binary_t&& bint, std::uint8_t st) noexcept(noexcept(binary_t(std::move(bint))))
|
wrapped_binary_t(binary_t&& b, std::uint8_t subtype) noexcept(noexcept(binary_t(std::move(b))))
|
||||||
: binary_t(std::move(bint))
|
: binary_t(std::move(b))
|
||||||
, m_subtype(st)
|
, m_subtype(subtype)
|
||||||
, m_has_subtype(true)
|
, m_has_subtype(true)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
|
@ -16545,6 +16569,21 @@ class basic_json
|
||||||
While @ref binary_t is used to define how binary values are stored, this
|
While @ref binary_t is used to define how binary values are stored, this
|
||||||
type is used to access binary values once they are parsed.
|
type is used to access binary values once they are parsed.
|
||||||
|
|
||||||
|
Notes on subtypes:
|
||||||
|
|
||||||
|
- CBOR
|
||||||
|
- Binary values are represented as byte strings. No subtypes are
|
||||||
|
supported and will be ignored when CBOR is written.
|
||||||
|
- MessagePack
|
||||||
|
- If a subtype is given and the binary array contains exactly 1, 2, 4, 8,
|
||||||
|
or 16 elements, the fixext family (fixext1, fixext2, fixext4, fixext8)
|
||||||
|
is used. For other sizes, the ext family (ext8, ext16, ext32) is used.
|
||||||
|
The subtype is then added as singed 8-bit integer.
|
||||||
|
- If no subtype is given, the bin family (bin8, bin16, bin32) is used.
|
||||||
|
- BSON
|
||||||
|
- If a subtype is given, it is used and added as unsigned 8-bit integer.
|
||||||
|
- If no subtype is given, the generic binary subtype 0x00 is used.
|
||||||
|
|
||||||
@sa @ref binary_array -- create a binary array
|
@sa @ref binary_array -- create a binary array
|
||||||
*/
|
*/
|
||||||
using internal_binary_t = nlohmann::detail::wrapped_binary_t<BinaryType>;
|
using internal_binary_t = nlohmann::detail::wrapped_binary_t<BinaryType>;
|
||||||
|
|
|
@ -77,7 +77,7 @@ class SaxEventLogger
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool binary(std::vector<std::uint8_t>& val)
|
bool binary(json::internal_binary_t& val)
|
||||||
{
|
{
|
||||||
std::string binary_contents = "binary(";
|
std::string binary_contents = "binary(";
|
||||||
std::string comma_space = "";
|
std::string comma_space = "";
|
||||||
|
@ -183,7 +183,7 @@ class SaxCountdown : public nlohmann::json::json_sax_t
|
||||||
return events_left-- > 0;
|
return events_left-- > 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool binary(std::vector<std::uint8_t>&) override
|
bool binary(json::internal_binary_t&) override
|
||||||
{
|
{
|
||||||
return events_left-- > 0;
|
return events_left-- > 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -76,7 +76,7 @@ struct SaxEventLogger : public nlohmann::json_sax<json>
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool binary(std::vector<std::uint8_t>& val) override
|
bool binary(json::internal_binary_t& val) override
|
||||||
{
|
{
|
||||||
std::string binary_contents = "binary(";
|
std::string binary_contents = "binary(";
|
||||||
std::string comma_space = "";
|
std::string comma_space = "";
|
||||||
|
|
Loading…
Reference in a new issue