33 namespace json_details {
47 template<
typename JsonMember,
typename ParseState>
48 [[nodiscard]] DAW_ATTRIB_INLINE
static constexpr json_result_t<JsonMember>
49 parse_ordered_class_member( std::size_t &member_index,
50 ParseState &parse_state ) {
52 using json_member_t = ordered_member_subtype_t<JsonMember>;
54 parse_state.move_next_member_or_end( );
59 if constexpr( is_an_ordered_member_v<JsonMember> ) {
60 pocm_details::maybe_skip_members<is_json_nullable_v<json_member_t>>(
61 parse_state, member_index, JsonMember::member_index );
64 ErrorReason::UnexpectedEndOfData,
71 if( DAW_UNLIKELY( parse_state.front( ) ==
']' ) ) {
72 if constexpr( is_json_nullable_v<json_member_t> ) {
74 auto loc = ParseState{ };
75 return parse_value<json_member_t,
77 json_member_t::expected_type>( loc );
80 true, missing_member(
"ordered_class_member" ), parse_state );
83 return parse_value<json_member_t, false, json_member_t::expected_type>(
97 template<
typename JsonMember,
bool NeedsClassPositions,
99 [[nodiscard]]
constexpr json_result_t<JsonMember>
100 parse_class_member_impl( ParseState &parse_state,
101 find_result<ParseState> fr ) {
103 auto [loc, known] = fr;
107 if constexpr( NeedsClassPositions ) {
108 auto const cf = parse_state.class_first;
109 auto const cl = parse_state.class_last;
111 json_result_t<without_name<JsonMember>>> ) {
112 auto const after_parse = daw::on_scope_exit( [&] {
113 parse_state.class_first = cf;
114 parse_state.class_last = cl;
116 return parse_value<without_name<JsonMember>,
118 JsonMember::expected_type>( parse_state );
121 parse_value<without_name<JsonMember>,
123 JsonMember::expected_type>( parse_state );
124 parse_state.class_first = cf;
125 parse_state.class_last = cl;
129 return parse_value<without_name<JsonMember>,
131 JsonMember::expected_type>( parse_state );
135 if( loc.is_null( ) ) {
136 if constexpr( is_json_nullable_v<JsonMember> ) {
137 return parse_value_null<without_name<JsonMember>,
true>( loc );
140 missing_member( std::string_view(
141 std::data( JsonMember::name ),
142 std::size( JsonMember::name ) ) ),
148 return parse_value<without_name<JsonMember>,
150 JsonMember::expected_type>( loc );
153 template<std::size_t member_position,
typename JsonMember,
154 AllMembersMustExist must_exist,
bool NeedsClassPositions,
155 typename ParseState, std::size_t N,
bool B>
156 [[nodiscard]] DAW_ATTRIB_INLINE
constexpr json_result_t<JsonMember>
157 parse_class_member( ParseState &parse_state,
158 locations_info_t<N, B> &locations ) {
159 parse_state.move_next_member_or_end( );
162 parse_state.is_at_next_class_member( ),
163 ErrorReason::MissingMemberNameOrEndOfClass,
166 return parse_class_member_impl<JsonMember, NeedsClassPositions>(
168 find_class_member<member_position, must_exist>(
171 is_json_nullable_v<JsonMember>,
172 JsonMember::name ) );
175 template<
bool IsExactClass,
typename ParseState,
typename OldClassPos>
176 DAW_ATTRIB_INLINE
static constexpr void
177 class_cleanup_now( ParseState &parse_state,
178 OldClassPos
const &old_class_pos ) {
180 ErrorReason::UnexpectedEndOfData,
182 parse_state.move_next_member_or_end( );
184 parse_state.move_to_next_class_member( );
185 if constexpr( IsExactClass ) {
187 ErrorReason::UnknownMember,
189 parse_state.remove_prefix( );
191 (void)parse_state.skip_class( );
195 parse_state.trim_left_checked( );
196 parse_state.set_class_position( old_class_pos );
206 template<
typename JsonClass,
typename... JsonMembers,
typename ParseState,
208 [[nodiscard]] DAW_ATTRIB_INLINE
constexpr json_result_t<JsonClass>
209 parse_json_class( ParseState &parse_state, std::index_sequence<Is...> ) {
210 static_assert( is_a_json_type_v<JsonClass> );
211 using T = json_result_t<JsonClass>;
212 using Constructor = json_constructor_t<JsonClass>;
213 static_assert( has_json_data_contract_trait_v<T>,
"Unexpected type" );
215 daw::constant<( all_json_members_must_exist_v<T, ParseState>
216 ? AllMembersMustExist::yes
217 : AllMembersMustExist::no )>;
219 parse_state.trim_left( );
222 ErrorReason::InvalidClassStart,
225 auto const old_class_pos = parse_state.get_class_position( );
226 parse_state.set_class_position( );
227 parse_state.remove_prefix( );
228 parse_state.trim_left( );
230 if constexpr(
sizeof...( JsonMembers ) == 0 ) {
232 class_cleanup_now<all_json_members_must_exist_v<T, ParseState>>(
233 parse_state, old_class_pos );
240 return construct_value_tp<T, Constructor>( parse_state,
244 using NeedClassPositions = std::bool_constant<(
245 ( must_be_class_member_v<typename JsonMembers::without_name> or
248#if defined( DAW_JSON_BUGFIX_MSVC_KNOWN_LOC_ICE_003 )
249 auto known_locations =
250 make_locations_info<ParseState, JsonMembers...>( );
252 auto known_locations = DAW_AS_CONSTANT(
253 ( make_locations_info<ParseState, JsonMembers...>( ) ) );
256 if constexpr( is_pinned_type_v<json_result_t<JsonClass>> ) {
260 auto const run_after_parse = daw::on_exit_success( [&] {
261 class_cleanup_now<all_json_members_must_exist_v<T, ParseState>>(
262 parse_state, old_class_pos );
264 (void)run_after_parse;
270 parse_class_member<Is,
271 daw::traits::nth_element<Is, JsonMembers...>,
273 NeedClassPositions::value>(
274 parse_state, known_locations )... };
276 return construct_value_tp<T, Constructor>(
278 fwd_pack{ parse_class_member<
280 daw::traits::nth_element<Is, JsonMembers...>,
282 NeedClassPositions::value>( parse_state,
283 known_locations )... } );
290 parse_class_member<Is,
291 daw::traits::nth_element<Is, JsonMembers...>,
293 NeedClassPositions::value>(
294 parse_state, known_locations )... };
296 class_cleanup_now<all_json_members_must_exist_v<T, ParseState>>(
297 parse_state, old_class_pos );
300 auto result = construct_value_tp<T, Constructor>(
302 fwd_pack{ parse_class_member<
304 daw::traits::nth_element<Is, JsonMembers...>,
306 NeedClassPositions::value>( parse_state,
307 known_locations )... } );
309 class_cleanup_now<all_json_members_must_exist_v<T, ParseState>>(
310 parse_state, old_class_pos );
316#if defined( DAW_JSON_HAS_REFLECTION )
317 template<
typename JsonClass,
typename... JsonMembers,
typename ParseState,
319 [[nodiscard]] DAW_ATTRIB_INLINE
constexpr json_result_t<JsonClass>
320 parse_json_reflected_class( ParseState &parse_state,
321 std::index_sequence<Is...> ) {
322 static_assert(
sizeof...( JsonMembers ) > 0 );
323 using T = json_result_t<JsonClass>;
324 static_assert( refl_details::PotentiallyReflectable<T> );
326 static constexpr auto must_exist =
327 all_json_members_must_exist_v<T, ParseState>
328 ? AllMembersMustExist::yes
329 : AllMembersMustExist::no;
331 parse_state.trim_left( );
334 ErrorReason::InvalidClassStart,
337 auto const old_class_pos = parse_state.get_class_position( );
338 parse_state.set_class_position( );
339 parse_state.remove_prefix( );
340 parse_state.trim_left( );
342 static constexpr auto NeedClassPositions =
343 ( must_be_class_member_v<typename JsonMembers::without_name> or ... );
345 auto known_locations =
346 make_locations_info<ParseState, JsonMembers...>( );
348 auto result = T{ parse_class_member<Is,
352 parse_state, known_locations )... };
354 class_cleanup_now<all_json_members_must_exist_v<T, ParseState>>(
355 parse_state, old_class_pos );
364 template<
typename JsonClass,
typename... JsonMembers,
typename ParseState>
365 [[nodiscard]]
static constexpr json_result_t<JsonClass>
366 parse_json_tuple_class( ParseState &parse_state ) {
367 static_assert( is_a_json_type_v<JsonClass> );
368 using T = json_base_type_t<JsonClass>;
369 using Constructor = json_constructor_t<JsonClass>;
370 static_assert( has_json_data_contract_trait_v<T>,
"Unexpected type" );
372 daw::is_callable_v<Constructor, json_result_t<JsonMembers>...>,
373 "Supplied types cannot be used for construction of this type" );
375 parse_state.trim_left( );
377 ErrorReason::InvalidArrayStart,
379 auto const old_class_pos = parse_state.get_class_position( );
380 parse_state.set_class_position( );
381 parse_state.remove_prefix( );
382 parse_state.trim_left( );
384 std::size_t current_idx = 0;
386 if constexpr( is_pinned_type_v<json_result_t<JsonClass>> ) {
387 auto const run_after_parse = daw::on_exit_success( [&] {
388 ordered_class_cleanup<all_json_members_must_exist_v<T, ParseState>,
390 decltype( old_class_pos )>( parse_state,
393 (void)run_after_parse;
397 return T{ parse_ordered_class_member<JsonMembers>(
398 current_idx, parse_state )... };
400 return construct_value_tp<T, Constructor>(
402 fwd_pack{ parse_ordered_class_member<JsonMembers>(
403 current_idx, parse_state )... } );
410 return T{ parse_ordered_class_member<JsonMembers>(
411 current_idx, parse_state )... };
413 return construct_value_tp<T, Constructor>(
415 fwd_pack{ parse_ordered_class_member<JsonMembers>(
416 current_idx, parse_state )... } );
419 if constexpr( all_json_members_must_exist_v<T, ParseState> ) {
420 parse_state.trim_left( );
422 ErrorReason::UnknownMember,
424 parse_state.remove_prefix( );
425 parse_state.trim_left( );
427 (void)parse_state.skip_array( );
429 parse_state.set_class_position( old_class_pos );