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 );
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 std::data( JsonMember::name ),
141 std::size( JsonMember::name ) ) ),
147 return parse_value<without_name<JsonMember>,
149 JsonMember::expected_type>( loc );
152 template<std::size_t member_position,
typename JsonMember,
153 AllMembersMustExist must_exist,
bool NeedsClassPositions,
154 typename ParseState, std::size_t N,
bool B>
155 [[nodiscard]] DAW_ATTRIB_INLINE
constexpr json_result_t<JsonMember>
156 parse_class_member( ParseState &parse_state,
157 locations_info_t<N, B> &locations ) {
158 parse_state.move_next_member_or_end( );
161 parse_state.is_at_next_class_member( ),
162 ErrorReason::MissingMemberNameOrEndOfClass,
165 return parse_class_member_impl<JsonMember, NeedsClassPositions>(
167 find_class_member<member_position, must_exist>(
170 is_json_nullable_v<JsonMember>,
171 JsonMember::name ) );
174 template<
bool IsExactClass,
typename ParseState,
typename OldClassPos>
175 DAW_ATTRIB_INLINE
static constexpr void
176 class_cleanup_now( ParseState &parse_state,
177 OldClassPos
const &old_class_pos ) {
179 ErrorReason::UnexpectedEndOfData,
181 parse_state.move_next_member_or_end( );
183 parse_state.move_to_next_class_member( );
184 if constexpr( IsExactClass ) {
186 ErrorReason::UnknownMember,
188 parse_state.remove_prefix( );
190 (void)parse_state.skip_class( );
194 parse_state.trim_left_checked( );
195 parse_state.set_class_position( old_class_pos );
205 template<
typename JsonClass,
typename... JsonMembers,
typename ParseState,
207 [[nodiscard]] DAW_ATTRIB_INLINE
constexpr json_result_t<JsonClass>
208 parse_json_class( ParseState &parse_state, std::index_sequence<Is...> ) {
209 static_assert( is_a_json_type_v<JsonClass> );
210 using T = json_result_t<JsonClass>;
211 using Constructor = json_constructor_t<JsonClass>;
212 static_assert( has_json_data_contract_trait_v<T>,
"Unexpected type" );
214 daw::constant<( all_json_members_must_exist_v<T, ParseState>
215 ? AllMembersMustExist::yes
216 : AllMembersMustExist::no )>;
218 parse_state.trim_left( );
221 ErrorReason::InvalidClassStart,
224 auto const old_class_pos = parse_state.get_class_position( );
225 parse_state.set_class_position( );
226 parse_state.remove_prefix( );
227 parse_state.trim_left( );
229 if constexpr(
sizeof...( JsonMembers ) == 0 ) {
231 class_cleanup_now<all_json_members_must_exist_v<T, ParseState>>(
232 parse_state, old_class_pos );
239 return construct_value_tp<T, Constructor>( parse_state,
243 using NeedClassPositions = std::bool_constant<(
244 ( must_be_class_member_v<typename JsonMembers::without_name> or
247#if defined( DAW_JSON_BUGFIX_MSVC_KNOWN_LOC_ICE_003 )
248 auto known_locations =
249 make_locations_info<ParseState, JsonMembers...>( );
251 auto known_locations = DAW_AS_CONSTANT(
252 ( make_locations_info<ParseState, JsonMembers...>( ) ) );
255 if constexpr( is_pinned_type_v<json_result_t<JsonClass>> ) {
259 auto const run_after_parse = daw::on_exit_success( [&] {
260 class_cleanup_now<all_json_members_must_exist_v<T, ParseState>>(
261 parse_state, old_class_pos );
263 (void)run_after_parse;
269 parse_class_member<Is,
270 daw::traits::nth_element<Is, JsonMembers...>,
272 NeedClassPositions::value>(
273 parse_state, known_locations )... };
275 return construct_value_tp<T, Constructor>(
277 fwd_pack{ parse_class_member<
279 daw::traits::nth_element<Is, JsonMembers...>,
281 NeedClassPositions::value>( parse_state,
282 known_locations )... } );
289 parse_class_member<Is,
290 daw::traits::nth_element<Is, JsonMembers...>,
292 NeedClassPositions::value>(
293 parse_state, known_locations )... };
295 class_cleanup_now<all_json_members_must_exist_v<T, ParseState>>(
296 parse_state, old_class_pos );
299 auto result = construct_value_tp<T, Constructor>(
301 fwd_pack{ parse_class_member<
303 daw::traits::nth_element<Is, JsonMembers...>,
305 NeedClassPositions::value>( parse_state,
306 known_locations )... } );
308 class_cleanup_now<all_json_members_must_exist_v<T, ParseState>>(
309 parse_state, old_class_pos );
321 template<
typename JsonClass,
typename... JsonMembers,
typename ParseState>
322 [[nodiscard]]
static constexpr json_result_t<JsonClass>
323 parse_json_tuple_class( ParseState &parse_state ) {
324 static_assert( is_a_json_type_v<JsonClass> );
325 using T = json_base_type_t<JsonClass>;
326 using Constructor = json_constructor_t<JsonClass>;
327 static_assert( has_json_data_contract_trait_v<T>,
"Unexpected type" );
329 daw::is_callable_v<Constructor, json_result_t<JsonMembers>...>,
330 "Supplied types cannot be used for construction of this type" );
332 parse_state.trim_left( );
334 ErrorReason::InvalidArrayStart,
336 auto const old_class_pos = parse_state.get_class_position( );
337 parse_state.set_class_position( );
338 parse_state.remove_prefix( );
339 parse_state.trim_left( );
341 std::size_t current_idx = 0;
343 if constexpr( is_pinned_type_v<json_result_t<JsonClass>> ) {
344 auto const run_after_parse = daw::on_exit_success( [&] {
345 ordered_class_cleanup<all_json_members_must_exist_v<T, ParseState>,
347 decltype( old_class_pos )>( parse_state,
350 (void)run_after_parse;
354 return T{ parse_ordered_class_member<JsonMembers>(
355 current_idx, parse_state )... };
357 return construct_value_tp<T, Constructor>(
359 fwd_pack{ parse_ordered_class_member<JsonMembers>(
360 current_idx, parse_state )... } );
367 return T{ parse_ordered_class_member<JsonMembers>(
368 current_idx, parse_state )... };
370 return construct_value_tp<T, Constructor>(
372 fwd_pack{ parse_ordered_class_member<JsonMembers>(
373 current_idx, parse_state )... } );
376 if constexpr( all_json_members_must_exist_v<T, ParseState> ) {
377 parse_state.trim_left( );
379 ErrorReason::UnknownMember,
381 parse_state.remove_prefix( );
382 parse_state.trim_left( );
384 (void)parse_state.skip_array( );
386 parse_state.set_class_position( old_class_pos );