24#include <daw/daw_algorithm.h>
25#include <daw/daw_move.h>
26#include <daw/daw_utility.h>
30#include <daw/stdinc/tuple_traits.h>
38 template<
json_options_t PolicyFlags = json_details::default_policy_flag,
39 typename Allocator = json_details::NoAllocator>
44 std::optional<std::string_view>
name;
48 template<std::
size_t Idx, json_options_t PolicyFlags,
typename Allocator>
49 constexpr decltype( auto )
53 "Invalid index. Valid values are 0 for name, and 1 for value" );
54 if constexpr( Idx == 0 ) {
55 return parse_state.
name;
57 return parse_state.
value;
61 template<std::
size_t Idx, json_options_t PolicyFlags,
typename Allocator>
62 constexpr decltype( auto )
66 "Invalid index. Valid values are 0 for name, and 1 for value" );
67 if constexpr( Idx == 0 ) {
68 return parse_state.
name;
70 return parse_state.
value;
74 template<std::
size_t Idx, json_options_t PolicyFlags,
typename Allocator>
75 constexpr decltype( auto )
79 "Invalid index. Valid values are 0 for name, and 1 for value" );
80 if constexpr( Idx == 0 ) {
81 return std::move( parse_state.
name );
83 return std::move( parse_state.
value );
90 template<daw::json::json_options_t PolicyFlags,
typename Allocator>
91 class tuple_element<0,
daw::json::basic_json_pair<PolicyFlags, Allocator>> {
93 using type = std::optional<std::string_view>;
96 template<daw::json::json_options_t PolicyFlags,
typename Allocator>
97 class tuple_element<1,
daw::json::basic_json_pair<PolicyFlags, Allocator>> {
102 template<daw::json::json_options_t PolicyFlags,
typename Allocator>
103 inline constexpr std::size_t
104 tuple_size_v<daw::json::basic_json_pair<PolicyFlags, Allocator>> = 2;
106 template<daw::json::json_options_t PolicyFlags,
typename Allocator>
107 class tuple_size<
daw::json::basic_json_pair<PolicyFlags, Allocator>> {
108 static constexpr std::size_t value = 2;
117 template<json_options_t PolicyFlags = json_details::default_policy_flag,
118 typename Allocator = json_details::NoAllocator>
126 using pointer = json_details::arrow_proxy<value_type>;
141 : m_state( parse_state ) {}
144 : m_state(
std::data( json_doc ),
daw::data_end( json_doc ) ) {}
147 Allocator
const &alloc )
148 : m_state(
std::data( json_doc ),
daw::data_end( json_doc ),
149 std::data( json_doc ),
daw::data_end( json_doc ), alloc ) {}
153 : m_state( jv.get_raw_state( ) ) {}
157 [[nodiscard]]
constexpr std::optional<std::string_view>
name( )
const {
161 auto parse_state = m_state;
162 auto result = json_details::parse_name( parse_state );
163 return std::string_view( std::data( result ), std::size( result ) );
174 auto parse_state = m_state;
175 (void)json_details::parse_name( parse_state );
180 parse_state.get_allocator( ) );
193 m_state.get_allocator( ) ) ) };
195 auto parse_state = m_state;
196 auto name = json_details::parse_name( parse_state );
198 std::string_view( std::data( name ), std::size( name ) ),
203 parse_state.get_allocator( ) ) ) };
211 return { operator*( ) };
219 (void)json_details::parse_name( m_state );
221 (void)json_details::skip_value( m_state );
222 m_state.move_next_member_or_end( );
235 return *m_state.class_first ==
'[';
241 return *m_state.class_first ==
'{';
246 [[nodiscard]]
constexpr bool good( )
const {
247 if( m_state.is_null( ) or not m_state.has_more( ) ) {
250 switch( m_state.front( ) ) {
280 [[nodiscard]]
constexpr explicit operator bool( )
const {
293 template<json_options_t P,
typename A>
294 [[nodiscard]]
constexpr bool
298 return m_state.first == rhs.
m_state.first;
302 return not rhs.
good( );
308 template<json_options_t P,
typename A>
309 [[nodiscard]]
constexpr bool
311 return not operator==( rhs );
315 template<json_options_t PolicyFlags,
typename Allocator>
323 template<
typename Allocator>
328 template<json_options_t PolicyFlags,
typename Allocator>
335 template<
json_options_t PolicyFlags = json_details::default_policy_flag,
336 typename Allocator = json_details::NoAllocator>
350 template<json_options_t PolicyFlags,
typename Allocator>
359 template<json_options_t PolicyFlags,
typename Allocator>
373 template<json_options_t P,
typename A>
375 : m_parse_state(
std::move( parse_state ) ) {
377 m_parse_state.trim_left( );
382 : m_parse_state(
std::data( sv ),
daw::data_end( sv ) ) {
383 m_parse_state.trim_left( );
388 : m_parse_state( first, first + static_cast<
std::ptrdiff_t>( sz ) ) {
389 m_parse_state.trim_left( );
394 : m_parse_state( first, last ) {
395 m_parse_state.trim_left( );
401 return m_parse_state;
405 return std::string_view( m_parse_state.first, m_parse_state.size( ) );
412 auto parse_state =
ParseState( m_parse_state.first,
416 m_parse_state.get_allocator( ) );
417 parse_state.remove_prefix( );
418 parse_state.trim_left( );
434 if( type( ) != JsonBaseParseTypes::Class ) {
437 bool const has_escape = name.contains(
'\\' );
440 return daw::algorithm::find_if(
441 begin( ), end( ), [name](
auto const &jp ) {
443 auto f0 = std::data( name );
444 auto const l0 = daw::data_end( name );
445 auto f1 = std::data( *jp.name );
446 auto const l1 = daw::data_end( *jp.name );
447 while( f0 != l0 and f1 != l1 ) {
458 return f0 == l0 and f1 == l1;
461 return daw::algorithm::find_if(
462 begin( ), end( ), [name](
auto const &jp ) {
464 return jp.name == name;
469 if( pos == end( ) ) {
472 return ( *pos ).value;
480 while( not json_path.empty( ) and jv ) {
482 if( json_path.front( ) ==
'[' ) {
483 return json_path.pop_front_until(
']' );
485 return json_path.pop_front_until( escaped_any_of<
'.',
'['>{ },
488 if( not json_path.empty( ) and json_path.front( ) ==
'.' ) {
489 json_path.remove_prefix( );
491 if( member.front( ) ==
'[' ) {
492 member.remove_prefix( );
495 std::data( member ), daw::data_end( member ) )
496 .with_allocator( m_parse_state.get_allocator( ) );
498 json_details::unsigned_parser<std::size_t,
499 options::JsonRangeCheck::Never,
503 if( not json_path.empty( ) and json_path.front( ) ==
'.' ) {
504 json_path.remove_prefix( );
515 template<
typename Result>
516 [[nodiscard]]
constexpr auto as( )
const {
517 using result_t = json_details::json_deduced_type<Result>;
518 auto state = m_parse_state;
519 return json_details::
520 parse_value<result_t, false, result_t::expected_type>( state );
523 template<
typename Result>
524 [[nodiscard]]
explicit operator Result( )
const {
525 return as<Result>( );
534 return find_member( json_path );
542 auto first = begin( );
543 auto const last = end( );
544 while( nsc_and( index > 0, first != last ) ) {
549 return ( *first ).value;
558 assert( type( ) == JsonBaseParseTypes::Array );
559 return find_element( index );
567 return find_element( index );
574 if( m_parse_state.empty( ) ) {
575 return JsonBaseParseTypes::None;
577 switch( m_parse_state.front( ) ) {
579 return JsonBaseParseTypes::String;
581 return JsonBaseParseTypes::Class;
583 return JsonBaseParseTypes::Array;
595 return JsonBaseParseTypes::Number;
597 if constexpr( not ParseState::is_unchecked_input ) {
598 if( m_parse_state.starts_with(
"true" ) ) {
599 return JsonBaseParseTypes::Bool;
601 return JsonBaseParseTypes::None;
603 return JsonBaseParseTypes::Bool;
606 if constexpr( not ParseState::is_unchecked_input ) {
607 if( m_parse_state.starts_with(
"false" ) ) {
608 return JsonBaseParseTypes::Bool;
610 return JsonBaseParseTypes::None;
612 return JsonBaseParseTypes::Bool;
616 ErrorReason::InvalidNull,
618 return JsonBaseParseTypes::Null;
620 return JsonBaseParseTypes::None;
626 auto parse_state = m_parse_state;
627 auto result = json_details::skip_value( parse_state );
639 auto parse_state = m_parse_state;
640 auto result = json_details::skip_value( parse_state );
641 return { std::data( result ), std::size( result ) };
647 template<
typename Alloc = std::allocator<
char>,
648 typename Traits = std::
char_traits<
char>>
649 [[nodiscard]] std::basic_string<char, Traits, Alloc>
651 auto parse_state = m_parse_state;
652 auto result = json_details::skip_value( parse_state );
653 return { std::data( result ), std::size( result ), alloc };
658 [[nodiscard]]
constexpr bool is_null( )
const {
659 return type( ) == JsonBaseParseTypes::Null;
665 return type( ) == JsonBaseParseTypes::Class;
671 return type( ) == JsonBaseParseTypes::Array;
677 return type( ) == JsonBaseParseTypes::Number;
683 return type( ) == JsonBaseParseTypes::String;
688 [[nodiscard]]
constexpr bool is_bool( )
const {
689 return type( ) == JsonBaseParseTypes::Bool;
697 return type( ) == JsonBaseParseTypes::None;
701 template<json_options_t P,
typename A>
702 [[nodiscard]]
constexpr
706 new_range.class_first = m_parse_state.class_first;
707 new_range.class_last = m_parse_state.class_last;
712 [[nodiscard]]
explicit constexpr operator bool( )
const {
713 return type( ) != JsonBaseParseTypes::None;
717 template<json_options_t PolicyFlags,
typename Allocator>
728 template<
typename Result, json_options_t PolicyFlags,
typename Allocator>
729 [[nodiscard]]
constexpr Result
731 return jv.template as<Result>( );
734 namespace json_details {
737 inline constexpr bool is_json_value =
false;
739 template<json_options_t PolicyFlags,
typename Allocator>
740 inline constexpr bool
741 is_json_value<basic_json_value<PolicyFlags, Allocator>> =
true;
743 template<json_options_t PolicyFlags,
typename Allocator>
744 inline constexpr bool
745 is_string_view_like_v<basic_json_value<PolicyFlags, Allocator>> =
false;
std::optional< std::string_view > type
#define daw_json_assert_weak(Bool,...)
Assert that Bool is true when in Checked Input mode If false pass rest of args to daw_json_error.
DAW_ATTRIB_NOINLINE void daw_json_error(ErrorReason reason)
JsonBaseParseTypes
The fundamental JSON types.
std::uint32_t json_options_t
daw::conditional_t< ParsePolicy::is_default_parse_policy, DefaultParsePolicy, ParsePolicy > TryDefaultParsePolicy
constexpr Result as(basic_json_value< PolicyFlags, Allocator > const &jv)
constexpr decltype(auto) get(basic_json_pair< PolicyFlags, Allocator > const &parse_state)
Customization point traits.
Handles the bounds and policy items for parsing execution and comments.
A name/value pair of string_view/json_value.
basic_json_value< PolicyFlags, Allocator > value
TryDefaultParsePolicy< BasicParsePolicy< PolicyFlags, Allocator > > ParseState
std::optional< std::string_view > name
a rudimentary range object for holding basic_json_value_iterator
constexpr iterator begin()
Iterator for iterating over arbitrary JSON members and array elements.
constexpr pointer operator->()
Return an arrow_proxy object containing the result of operator* Should not use this method unless you...
constexpr void operator++(int) &
Move parser to next value.
constexpr std::optional< std::string_view > name() const
Name of member.
constexpr basic_json_value_iterator(parse_policy const &parse_state)
constexpr parse_policy const & get_raw_state() const
Get access to the internal state. Should not be used as part of public API.
TryDefaultParsePolicy< BasicParsePolicy< PolicyFlags, Allocator > > parse_policy
basic_json_value_iterator()=default
basic_json_value_iterator(daw::string_view json_doc)
constexpr basic_json_value< PolicyFlags, Allocator > value() const
Get the value currently being referenced.
std::forward_iterator_tag iterator_category
constexpr basic_json_value_iterator & operator++()
Move the parser to the next value.
std::string_view key_type
std::ptrdiff_t difference_type
constexpr bool is_class() const
Is the value this iterator iterates over an class.
constexpr bool is_array() const
Is the value this iterator iterates over an array.
constexpr bool good() const
Can we increment more.
json_details::arrow_proxy< value_type > pointer
constexpr bool operator==(basic_json_value_iterator< P, A > const &rhs) const
Check for equivalence with rhs iterator.
basic_json_value_iterator(daw::string_view json_doc, Allocator const &alloc)
basic_json_value_iterator(basic_json_value< PolicyFlags, Allocator > const &jv)
constexpr basic_json_pair< PolicyFlags, Allocator > operator*()
Get the name/value pair of the currently referenced element.
constexpr bool operator!=(basic_json_value_iterator< P, A > const &rhs) const
Check if rhs is not equivalent to self.
A non-owning container for arbitrary JSON values that allows movement/iteration through.
constexpr bool is_null() const
Is the JSON value a null literal.
constexpr bool is_class() const
Is the JSON value a class.
constexpr std::string_view get_string_view() const
Construct a string range of the current value. Strings start inside the quotes.
constexpr std::string_view get_raw_json_document() const
constexpr bool is_bool() const
Is the JSON value a boolean.
constexpr iterator end() const
End of range over class/arrays members/items.
constexpr basic_json_value(char const *first, std::size_t sz)
Construct from char const *, std::size_t.
std::ptrdiff_t difference_type
constexpr basic_json_value(char const *first, char const *last)
Construct from char const *, char const *.
constexpr basic_json_value operator[](daw::string_view json_path) const
Query the current class for a named member.
TryDefaultParsePolicy< BasicParsePolicy< PolicyFlags, Allocator > > ParseState
constexpr ParseState get_state() const
Construct a string range of the current value.
constexpr basic_json_value(daw::string_view sv)
Construct from string_view.
constexpr auto as() const
Parse the current json member as a Result. The Result type must be supported or mapped via a json_dat...
constexpr basic_json_value find_array_element(std::size_t index) const
Find the nth element of the current json array.
constexpr iterator begin() const
Get the first member/item.
constexpr basic_json_value find_element(std::size_t index) const
Find the nth element/submember of the current json array or class.
constexpr basic_json_value find_member(daw::string_view json_path) const
find a class member/array element as specified by the json_path
constexpr bool is_array() const
Is the JSON value a array.
constexpr bool is_string() const
Is the JSON value a string.
constexpr bool is_number() const
Is the JSON value a number literal.
constexpr ParseState get_raw_state() const
Get a copy of the underlying parse state.
std::basic_string< char, Traits, Alloc > get_string(Alloc const &alloc=Alloc()) const
Construct a string range of the current value. Strings start inside the quotes.
basic_json_value()=default
constexpr basic_json_value(BasicParsePolicy< P, A > parse_state)
Construct from IteratorRange.
constexpr bool is_unknown() const
Is the JSON data unrecognizable. JSON members will start with one of ",[,{,0,1,2,3,...
constexpr JsonBaseParseTypes type() const
Get the type of JSON value.
constexpr basic_json_value operator[](std::size_t index) const
Find the nth element/submember of the current json array or class.
constexpr basic_json_value find_class_member(daw::string_view name) const
Query the current class for a named member.
#define DAW_JSON_VER
The version string used in namespace definitions. Must be a valid namespace name.