🔨 optimized output format
This commit is contained in:
parent
fd04967676
commit
965a70e38d
2 changed files with 125 additions and 37 deletions
112
src/json.hpp
112
src/json.hpp
|
@ -6189,20 +6189,30 @@ class binary_writer
|
||||||
}
|
}
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
@brief[in] j JSON value to serialize
|
@param[in] j JSON value to serialize
|
||||||
|
@param[in] use_count whether to use '#' prefixes (optimized format)
|
||||||
|
@param[in] use_type whether to use '$' prefixes (optimized format)
|
||||||
|
@param[in] add_prefix whether prefixes need to be used for this value
|
||||||
*/
|
*/
|
||||||
void write_ubjson(const BasicJsonType& j, const bool use_count = false)
|
void write_ubjson(const BasicJsonType& j,
|
||||||
|
const bool use_count = false,
|
||||||
|
const bool use_type = false,
|
||||||
|
const bool add_prefix = true)
|
||||||
{
|
{
|
||||||
switch (j.type())
|
switch (j.type())
|
||||||
{
|
{
|
||||||
case value_t::null:
|
case value_t::null:
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('Z'));
|
oa->write_character(static_cast<CharType>('Z'));
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case value_t::boolean:
|
case value_t::boolean:
|
||||||
{
|
{
|
||||||
|
if (add_prefix)
|
||||||
oa->write_character(j.m_value.boolean
|
oa->write_character(j.m_value.boolean
|
||||||
? static_cast<CharType>('T')
|
? static_cast<CharType>('T')
|
||||||
: static_cast<CharType>('F'));
|
: static_cast<CharType>('F'));
|
||||||
|
@ -6211,26 +6221,29 @@ class binary_writer
|
||||||
|
|
||||||
case value_t::number_integer:
|
case value_t::number_integer:
|
||||||
{
|
{
|
||||||
write_number_with_ubjson_prefix(j.m_value.number_integer);
|
write_number_with_ubjson_prefix(j.m_value.number_integer, add_prefix);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case value_t::number_unsigned:
|
case value_t::number_unsigned:
|
||||||
{
|
{
|
||||||
write_number_with_ubjson_prefix(j.m_value.number_unsigned);
|
write_number_with_ubjson_prefix(j.m_value.number_unsigned, add_prefix);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case value_t::number_float:
|
case value_t::number_float:
|
||||||
{
|
{
|
||||||
write_number_with_ubjson_prefix(j.m_value.number_float);
|
write_number_with_ubjson_prefix(j.m_value.number_float, add_prefix);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case value_t::string:
|
case value_t::string:
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('S'));
|
oa->write_character(static_cast<CharType>('S'));
|
||||||
write_number_with_ubjson_prefix(j.m_value.string->size());
|
}
|
||||||
|
write_number_with_ubjson_prefix(j.m_value.string->size(), true);
|
||||||
oa->write_characters(
|
oa->write_characters(
|
||||||
reinterpret_cast<const CharType*>(j.m_value.string->c_str()),
|
reinterpret_cast<const CharType*>(j.m_value.string->c_str()),
|
||||||
j.m_value.string->size());
|
j.m_value.string->size());
|
||||||
|
@ -6238,18 +6251,39 @@ class binary_writer
|
||||||
}
|
}
|
||||||
|
|
||||||
case value_t::array:
|
case value_t::array:
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('['));
|
oa->write_character(static_cast<CharType>('['));
|
||||||
|
}
|
||||||
|
|
||||||
|
bool prefix_required = true;
|
||||||
|
if (use_type and not j.m_value.array->empty())
|
||||||
|
{
|
||||||
|
const char first_prefix = ubjson_prefix(j.front());
|
||||||
|
const bool same_prefix = std::all_of(j.begin() + 1, j.end(),
|
||||||
|
[this, first_prefix](const BasicJsonType & v)
|
||||||
|
{
|
||||||
|
return ubjson_prefix(v) == first_prefix;
|
||||||
|
});
|
||||||
|
|
||||||
|
if (same_prefix)
|
||||||
|
{
|
||||||
|
prefix_required = false;
|
||||||
|
oa->write_character(static_cast<CharType>('$'));
|
||||||
|
oa->write_character(static_cast<CharType>(first_prefix));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (use_count)
|
if (use_count)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('#'));
|
oa->write_character(static_cast<CharType>('#'));
|
||||||
write_number_with_ubjson_prefix(j.m_value.array->size());
|
write_number_with_ubjson_prefix(j.m_value.array->size(), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const auto& el : *j.m_value.array)
|
for (const auto& el : *j.m_value.array)
|
||||||
{
|
{
|
||||||
write_ubjson(el, use_count);
|
write_ubjson(el, use_count, use_type, prefix_required);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (not use_count)
|
if (not use_count)
|
||||||
|
@ -6261,22 +6295,43 @@ class binary_writer
|
||||||
}
|
}
|
||||||
|
|
||||||
case value_t::object:
|
case value_t::object:
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('{'));
|
oa->write_character(static_cast<CharType>('{'));
|
||||||
|
}
|
||||||
|
|
||||||
|
bool prefix_required = true;
|
||||||
|
if (use_type and not j.m_value.object->empty())
|
||||||
|
{
|
||||||
|
const char first_prefix = ubjson_prefix(j.front());
|
||||||
|
const bool same_prefix = std::all_of(j.begin(), j.end(),
|
||||||
|
[this, first_prefix](const BasicJsonType & v)
|
||||||
|
{
|
||||||
|
return ubjson_prefix(v) == first_prefix;
|
||||||
|
});
|
||||||
|
|
||||||
|
if (same_prefix)
|
||||||
|
{
|
||||||
|
prefix_required = false;
|
||||||
|
oa->write_character(static_cast<CharType>('$'));
|
||||||
|
oa->write_character(static_cast<CharType>(first_prefix));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (use_count)
|
if (use_count)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('#'));
|
oa->write_character(static_cast<CharType>('#'));
|
||||||
write_number_with_ubjson_prefix(j.m_value.object->size());
|
write_number_with_ubjson_prefix(j.m_value.object->size(), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const auto& el : *j.m_value.object)
|
for (const auto& el : *j.m_value.object)
|
||||||
{
|
{
|
||||||
write_number_with_ubjson_prefix(el.first.size());
|
write_number_with_ubjson_prefix(el.first.size(), true);
|
||||||
oa->write_characters(
|
oa->write_characters(
|
||||||
reinterpret_cast<const CharType*>(el.first.c_str()),
|
reinterpret_cast<const CharType*>(el.first.c_str()),
|
||||||
el.first.size());
|
el.first.size());
|
||||||
write_ubjson(el.second, use_count);
|
write_ubjson(el.second, use_count, use_type, prefix_required);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (not use_count)
|
if (not use_count)
|
||||||
|
@ -6320,38 +6375,56 @@ class binary_writer
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename NumberType>
|
template<typename NumberType>
|
||||||
void write_number_with_ubjson_prefix(const NumberType n)
|
void write_number_with_ubjson_prefix(const NumberType n, const bool add_prefix)
|
||||||
{
|
{
|
||||||
if (std::is_floating_point<NumberType>::value)
|
if (std::is_floating_point<NumberType>::value)
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('D')); // float64
|
oa->write_character(static_cast<CharType>('D')); // float64
|
||||||
|
}
|
||||||
write_number(n);
|
write_number(n);
|
||||||
}
|
}
|
||||||
else if (std::is_unsigned<NumberType>::value)
|
else if (std::is_unsigned<NumberType>::value)
|
||||||
{
|
{
|
||||||
if (n <= (std::numeric_limits<int8_t>::max)())
|
if (n <= (std::numeric_limits<int8_t>::max)())
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('i')); // uint8
|
oa->write_character(static_cast<CharType>('i')); // uint8
|
||||||
|
}
|
||||||
write_number(static_cast<uint8_t>(n));
|
write_number(static_cast<uint8_t>(n));
|
||||||
}
|
}
|
||||||
else if (n <= (std::numeric_limits<uint8_t>::max)())
|
else if (n <= (std::numeric_limits<uint8_t>::max)())
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('U')); // uint8
|
oa->write_character(static_cast<CharType>('U')); // uint8
|
||||||
|
}
|
||||||
write_number(static_cast<uint8_t>(n));
|
write_number(static_cast<uint8_t>(n));
|
||||||
}
|
}
|
||||||
else if (n <= (std::numeric_limits<int16_t>::max)())
|
else if (n <= (std::numeric_limits<int16_t>::max)())
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('I')); // int16
|
oa->write_character(static_cast<CharType>('I')); // int16
|
||||||
|
}
|
||||||
write_number(static_cast<int16_t>(n));
|
write_number(static_cast<int16_t>(n));
|
||||||
}
|
}
|
||||||
else if (n <= (std::numeric_limits<int32_t>::max)())
|
else if (n <= (std::numeric_limits<int32_t>::max)())
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('l')); // int32
|
oa->write_character(static_cast<CharType>('l')); // int32
|
||||||
|
}
|
||||||
write_number(static_cast<int32_t>(n));
|
write_number(static_cast<int32_t>(n));
|
||||||
}
|
}
|
||||||
else if (n <= (std::numeric_limits<int64_t>::max)())
|
else if (n <= (std::numeric_limits<int64_t>::max)())
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('L')); // int64
|
oa->write_character(static_cast<CharType>('L')); // int64
|
||||||
|
}
|
||||||
write_number(static_cast<int64_t>(n));
|
write_number(static_cast<int64_t>(n));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
@ -6363,28 +6436,43 @@ class binary_writer
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if ((std::numeric_limits<int8_t>::min)() <= n and n <= (std::numeric_limits<int8_t>::max)())
|
if ((std::numeric_limits<int8_t>::min)() <= n and n <= (std::numeric_limits<int8_t>::max)())
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('i')); // int8
|
oa->write_character(static_cast<CharType>('i')); // int8
|
||||||
|
}
|
||||||
write_number(static_cast<int8_t>(n));
|
write_number(static_cast<int8_t>(n));
|
||||||
}
|
}
|
||||||
else if ((std::numeric_limits<uint8_t>::min)() <= n and n <= (std::numeric_limits<uint8_t>::max)())
|
else if ((std::numeric_limits<uint8_t>::min)() <= n and n <= (std::numeric_limits<uint8_t>::max)())
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('U')); // uint8
|
oa->write_character(static_cast<CharType>('U')); // uint8
|
||||||
|
}
|
||||||
write_number(static_cast<uint8_t>(n));
|
write_number(static_cast<uint8_t>(n));
|
||||||
}
|
}
|
||||||
else if ((std::numeric_limits<int16_t>::min)() <= n and n <= (std::numeric_limits<int16_t>::max)())
|
else if ((std::numeric_limits<int16_t>::min)() <= n and n <= (std::numeric_limits<int16_t>::max)())
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('I')); // int16
|
oa->write_character(static_cast<CharType>('I')); // int16
|
||||||
|
}
|
||||||
write_number(static_cast<int16_t>(n));
|
write_number(static_cast<int16_t>(n));
|
||||||
}
|
}
|
||||||
else if ((std::numeric_limits<int32_t>::min)() <= n and n <= (std::numeric_limits<int32_t>::max)())
|
else if ((std::numeric_limits<int32_t>::min)() <= n and n <= (std::numeric_limits<int32_t>::max)())
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('l')); // int32
|
oa->write_character(static_cast<CharType>('l')); // int32
|
||||||
|
}
|
||||||
write_number(static_cast<int32_t>(n));
|
write_number(static_cast<int32_t>(n));
|
||||||
}
|
}
|
||||||
else if ((std::numeric_limits<int64_t>::min)() <= n and n <= (std::numeric_limits<int64_t>::max)())
|
else if ((std::numeric_limits<int64_t>::min)() <= n and n <= (std::numeric_limits<int64_t>::max)())
|
||||||
|
{
|
||||||
|
if (add_prefix)
|
||||||
{
|
{
|
||||||
oa->write_character(static_cast<CharType>('L')); // int64
|
oa->write_character(static_cast<CharType>('L')); // int64
|
||||||
|
}
|
||||||
write_number(static_cast<int64_t>(n));
|
write_number(static_cast<int64_t>(n));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
|
Loading…
Reference in a new issue