28 namespace json_details {
29 namespace name::name_parser {
34 template<
typename ParseState>
35 static constexpr void trim_end_of_name( ParseState &parse_state ) {
36 parse_state.trim_left( );
39 ErrorReason::InvalidMemberName, parse_state );
40 parse_state.remove_prefix( );
41 parse_state.trim_left( );
44 template<
typename ParseState>
45 [[nodiscard]] DAW_ATTRIB_INLINE
static constexpr daw::string_view
46 parse_nq( ParseState &parse_state ) {
47 if constexpr( ParseState::allow_escaped_names ) {
48 auto r = skip_string_nq( parse_state );
49 trim_end_of_name( parse_state );
50 return daw::string_view( std::data( r ), std::size( r ) );
52 char const *
const ptr = parse_state.first;
53 if constexpr( ParseState::is_unchecked_input ) {
54 parse_state.template move_to_next_of_unchecked<
'"'>( );
56 parse_state.template move_to_next_of_checked<
'"'>( );
59 *std::prev( parse_state.first ) !=
'\\',
60 ErrorReason::InvalidString, parse_state );
61 auto result = daw::string_view( ptr, parse_state.first );
62 parse_state.remove_prefix( );
63 trim_end_of_name( parse_state );
69 struct pop_json_path_result {
70 daw::string_view current{ };
76 [[nodiscard]]
static constexpr pop_json_path_result
77 pop_json_path( daw::string_view &path ) {
78 auto result = pop_json_path_result{ };
82 if( path.front( ) ==
'.' ) {
83 path.remove_prefix( );
86 path.pop_front_until( [&, in_escape =
false](
char c )
mutable {
98 result.found_char = c;
107 [[nodiscard]]
static constexpr bool
108 json_path_compare( daw::string_view json_path_item,
109 daw::string_view member_name ) {
110 if( json_path_item.front( ) ==
'\\' ) {
111 json_path_item.remove_prefix( );
113 while( not json_path_item.empty( ) and not member_name.empty( ) ) {
114 if( json_path_item.front( ) != member_name.front( ) ) {
117 json_path_item.remove_prefix( );
118 if( not json_path_item.empty( ) and
119 json_path_item.front( ) ==
'\\' ) {
120 json_path_item.remove_prefix( );
122 member_name.remove_prefix( );
124 return std::size( json_path_item ) == std::size( member_name );
127 template<
typename Result,
typename ForwardIterator>
128 constexpr Result parse_unsigned_int( ForwardIterator first,
129 ForwardIterator last ) {
130 std::size_t count = daw::digits<Result>;
135 for( ; first != last and count > 0; ++first, --count ) {
136 result *=
static_cast<Result
>( 10 );
138 static_cast<Result
>( *first ) -
static_cast<Result
>(
'0' );
149 template<
typename ParseState>
150 [[nodiscard]] DAW_ATTRIB_FLATTEN
static inline constexpr daw::string_view
151 parse_name( ParseState &parse_state ) {
153 ErrorReason::InvalidMemberName, parse_state );
154 parse_state.remove_prefix( );
155 return name::name_parser::parse_nq( parse_state );
158 template<
typename ParseState>
159 static constexpr bool find_range2( ParseState &parse_state,
160 daw::string_view path ) {
162 auto pop_result = pop_json_path( path );
163 while( not pop_result.current.empty( ) ) {
164 if( pop_result.found_char ==
']' ) {
167 ErrorReason::InvalidJSONPath, parse_state );
168 parse_state.remove_prefix( );
169 parse_state.trim_left_unchecked( );
170 auto idx = parse_unsigned_int<std::size_t>(
171 pop_result.current.data( ), pop_result.current.data_end( ) );
175 (void)skip_value( parse_state );
176 parse_state.trim_left_checked( );
177 if( ( idx > 0 ) & ( parse_state.has_more( ) and
178 ( parse_state.front( ) !=
',' ) ) ) {
181 parse_state.move_next_member_or_end( );
185 ErrorReason::InvalidJSONPath, parse_state );
186 parse_state.remove_prefix( );
187 parse_state.trim_left_unchecked( );
188 auto name = parse_name( parse_state );
189 while( not json_path_compare( pop_result.current, name ) ) {
190 (void)skip_value( parse_state );
191 parse_state.move_next_member_or_end( );
192 if( parse_state.empty( ) or parse_state.front( ) !=
'"' ) {
195 name = parse_name( parse_state );
198 pop_result = pop_json_path( path );
203 template<
typename ParsePolicy>
204 [[nodiscard]]
static constexpr find_result<ParsePolicy>
205 find_range( daw::string_view str, daw::string_view start_path ) {
207 ParsePolicy( std::data( str ), daw::data_end( str ) );
208 parse_state.trim_left_checked( );
210 if( parse_state.has_more( ) and not start_path.empty( ) ) {
211 found = find_range2( parse_state, start_path );
213 return find_result<ParsePolicy>{ parse_state, found };
216 template<
typename ParsePolicy,
typename Allocator>
217 [[nodiscard]]
static constexpr find_result<ParsePolicy>
218 find_range( daw::string_view str, daw::string_view start_path,
221 std::is_same_v<char const *, typename ParsePolicy::iterator>,
222 "Only char const * ranges are currently supported" );
223 auto parse_state = ParsePolicy::with_allocator(
224 std::data( str ), daw::data_end( str ), alloc );
225 parse_state.trim_left_checked( );
226 if( parse_state.has_more( ) and not start_path.empty( ) ) {
227 if( not find_range2( parse_state, start_path ) ) {
228 return find_result<ParsePolicy>{ parse_state,
false };
231 return find_result<ParsePolicy>{ parse_state,
true };