DAW JSON Link
Loading...
Searching...
No Matches
daw_json_parse_class.h
Go to the documentation of this file.
1// Copyright (c) Darrell Wright
2//
3// Distributed under the Boost Software License, Version 1.0. (See accompanying
4// file LICENSE or copy at http://www.boost.org/LICENSE_1_0.txt)
5//
6// Official repository: https://github.com/beached/daw_json_link
7//
8
9#pragma once
10
11#include "version.h"
12
20
21#include <daw/daw_callable.h>
22#include <daw/daw_consteval.h>
23#include <daw/daw_constinit.h>
24#include <daw/daw_likely.h>
25#include <daw/daw_traits.h>
26#include <daw/traits/daw_traits_nth_element.h>
27
28#include <cstddef>
29#include <type_traits>
30
31namespace daw::json {
32 inline namespace DAW_JSON_VER {
33 namespace json_details {
34
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 ) {
51
52 using json_member_t = ordered_member_subtype_t<JsonMember>;
53
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 );
62 } else {
63 daw_json_assert_weak( parse_state.has_more( ),
64 ErrorReason::UnexpectedEndOfData,
65 parse_state );
66 }
67
68 // this is an out value, get position ready
69 ++member_index;
70
71 if( DAW_UNLIKELY( parse_state.front( ) == ']' ) ) {
72 if constexpr( is_json_nullable_v<json_member_t> ) {
73
74 auto loc = ParseState{ };
75 return parse_value<json_member_t,
76 true,
77 json_member_t::expected_type>( loc );
78 } else {
79 daw_json_error( missing_member( "ordered_class_member" ),
80 parse_state );
81 }
82 }
83 return parse_value<json_member_t, false, json_member_t::expected_type>(
84 parse_state );
85 }
86
97 template<typename JsonMember, bool NeedsClassPositions,
98 typename ParseState>
99 [[nodiscard]] constexpr json_result_t<JsonMember>
100 parse_class_member_impl( ParseState &parse_state,
101 find_result<ParseState> fr ) {
102
103 auto [loc, known] = fr;
104
105 // If the member was found loc will have it's position
106 if( not known ) {
107 if constexpr( NeedsClassPositions ) {
108 auto const cf = parse_state.class_first;
109 auto const cl = parse_state.class_last;
110 if constexpr( is_pinned_type_v<
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;
115 } );
116 return parse_value<without_name<JsonMember>,
117 false,
118 JsonMember::expected_type>( parse_state );
119 } else {
120 auto result =
121 parse_value<without_name<JsonMember>,
122 false,
123 JsonMember::expected_type>( parse_state );
124 parse_state.class_first = cf;
125 parse_state.class_last = cl;
126 return result;
127 }
128 } else {
129 return parse_value<without_name<JsonMember>,
130 false,
131 JsonMember::expected_type>( parse_state );
132 }
133 }
134 // We cannot find the member, check if the member is nullable
135 if( loc.is_null( ) ) {
136 if constexpr( is_json_nullable_v<JsonMember> ) {
137 return parse_value_null<without_name<JsonMember>, true>( loc );
138 } else {
139 daw_json_error( missing_member( std::string_view(
140 std::data( JsonMember::name ),
141 std::size( JsonMember::name ) ) ),
142 parse_state );
143 }
144 }
145
146 // Member was previously skipped
147 return parse_value<without_name<JsonMember>,
148 true,
149 JsonMember::expected_type>( loc );
150 }
151
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( );
159
160 daw_json_assert_weak( not parse_state.empty( ) and
161 parse_state.is_at_next_class_member( ),
162 ErrorReason::MissingMemberNameOrEndOfClass,
163 parse_state );
164
165 return parse_class_member_impl<JsonMember, NeedsClassPositions>(
166 parse_state,
167 find_class_member<member_position, must_exist>(
168 parse_state,
169 locations,
170 is_json_nullable_v<JsonMember>,
171 JsonMember::name ) );
172 }
173
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 ) {
178 daw_json_assert_weak( parse_state.has_more( ),
179 ErrorReason::UnexpectedEndOfData,
180 parse_state );
181 parse_state.move_next_member_or_end( );
182 // If we fulfill the contract before all values are parses
183 parse_state.move_to_next_class_member( );
184 if constexpr( IsExactClass ) {
185 daw_json_assert_weak( parse_state.front( ) == '}',
186 ErrorReason::UnknownMember,
187 parse_state );
188 parse_state.remove_prefix( );
189 } else {
190 (void)parse_state.skip_class( );
191 // Yes this must be checked. We maybe at the end of document. After
192 // the 2nd try, give up
193 }
194 parse_state.trim_left_checked( );
195 parse_state.set_class_position( old_class_pos );
196 }
197
205 template<typename JsonClass, typename... JsonMembers, typename ParseState,
206 std::size_t... Is>
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" );
213 using must_exist =
214 daw::constant<( all_json_members_must_exist_v<T, ParseState>
215 ? AllMembersMustExist::yes
216 : AllMembersMustExist::no )>;
217
218 parse_state.trim_left( );
219 // TODO, use member name
220 daw_json_assert_weak( parse_state.is_opening_brace_checked( ),
221 ErrorReason::InvalidClassStart,
222 parse_state );
223
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( );
228
229 if constexpr( sizeof...( JsonMembers ) == 0 ) {
230 // Clang-CL with MSVC has issues if we don't do empties this way
231 class_cleanup_now<all_json_members_must_exist_v<T, ParseState>>(
232 parse_state, old_class_pos );
233
234 if constexpr( should_construct_explicitly_v<Constructor,
235 T,
236 ParseState> ) {
237 return T{ };
238 } else {
239 return construct_value_tp<T, Constructor>( parse_state,
240 fwd_pack{ } );
241 }
242 } else {
243 using NeedClassPositions = std::bool_constant<(
244 ( must_be_class_member_v<typename JsonMembers::without_name> or
245 ... ) )>;
246
247#if defined( DAW_JSON_BUGFIX_MSVC_KNOWN_LOC_ICE_003 )
248 auto known_locations =
249 make_locations_info<ParseState, JsonMembers...>( );
250#else
251 auto known_locations = DAW_AS_CONSTANT(
252 ( make_locations_info<ParseState, JsonMembers...>( ) ) );
253#endif
254
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 );
262 } );
263 (void)run_after_parse;
264
265 if constexpr( should_construct_explicitly_v<Constructor,
266 T,
267 ParseState> ) {
268 return T{
269 parse_class_member<Is,
270 daw::traits::nth_element<Is, JsonMembers...>,
271 must_exist::value,
272 NeedClassPositions::value>(
273 parse_state, known_locations )... };
274 } else {
275 return construct_value_tp<T, Constructor>(
276 parse_state,
277 fwd_pack{ parse_class_member<
278 Is,
279 daw::traits::nth_element<Is, JsonMembers...>,
280 must_exist::value,
281 NeedClassPositions::value>( parse_state,
282 known_locations )... } );
283 }
284 } else {
285 if constexpr( should_construct_explicitly_v<Constructor,
286 T,
287 ParseState> ) {
288 auto result = T{
289 parse_class_member<Is,
290 daw::traits::nth_element<Is, JsonMembers...>,
291 must_exist::value,
292 NeedClassPositions::value>(
293 parse_state, known_locations )... };
294
295 class_cleanup_now<all_json_members_must_exist_v<T, ParseState>>(
296 parse_state, old_class_pos );
297 return result;
298 } else {
299 auto result = construct_value_tp<T, Constructor>(
300 parse_state,
301 fwd_pack{ parse_class_member<
302 Is,
303 daw::traits::nth_element<Is, JsonMembers...>,
304 must_exist::value,
305 NeedClassPositions::value>( parse_state,
306 known_locations )... } );
307
308 class_cleanup_now<all_json_members_must_exist_v<T, ParseState>>(
309 parse_state, old_class_pos );
310 return result;
311 }
312 }
313 }
314 }
315
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" );
328 static_assert(
329 daw::is_callable_v<Constructor, json_result_t<JsonMembers>...>,
330 "Supplied types cannot be used for construction of this type" );
331
332 parse_state.trim_left( ); // Move to array start '['
333 daw_json_assert_weak( parse_state.is_opening_bracket_checked( ),
334 ErrorReason::InvalidArrayStart,
335 parse_state );
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( );
340
341 std::size_t current_idx = 0;
342
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>,
346 ParseState,
347 decltype( old_class_pos )>( parse_state,
348 old_class_pos );
349 } );
350 (void)run_after_parse;
351 if constexpr( should_construct_explicitly_v<Constructor,
352 T,
353 ParseState> ) {
354 return T{ parse_ordered_class_member<JsonMembers>(
355 current_idx, parse_state )... };
356 } else {
357 return construct_value_tp<T, Constructor>(
358 parse_state,
359 fwd_pack{ parse_ordered_class_member<JsonMembers>(
360 current_idx, parse_state )... } );
361 }
362 } else {
363 auto result = [&] {
364 if constexpr( should_construct_explicitly_v<Constructor,
365 T,
366 ParseState> ) {
367 return T{ parse_ordered_class_member<JsonMembers>(
368 current_idx, parse_state )... };
369 } else {
370 return construct_value_tp<T, Constructor>(
371 parse_state,
372 fwd_pack{ parse_ordered_class_member<JsonMembers>(
373 current_idx, parse_state )... } );
374 }
375 }( );
376 if constexpr( all_json_members_must_exist_v<T, ParseState> ) {
377 parse_state.trim_left( );
378 daw_json_assert_weak( parse_state.front( ) == ']',
379 ErrorReason::UnknownMember,
380 parse_state );
381 parse_state.remove_prefix( );
382 parse_state.trim_left( );
383 } else {
384 (void)parse_state.skip_array( );
385 }
386 parse_state.set_class_position( old_class_pos );
387 return result;
388 }
389 }
390 } // namespace json_details
391 } // namespace DAW_JSON_VER
392} // namespace daw::json
#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)
constexpr bool is_pinned_type_v
Is the type pinned in memory and unable to be copied/moved after construction(e.g....
Customization point traits.
#define DAW_JSON_VER
The version string used in namespace definitions. Must be a valid namespace name.
Definition version.h:20