-
Notifications
You must be signed in to change notification settings - Fork 98
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Handling optional-like sequences #1050
Comments
Today as described classes, because description is inherently opt-in, but tomorrow, when reflection arrives, the answer may need to change. |
That's not the case right now: template <bool B>
struct X {
int m;
// admittedly this is silly, but it's just an example
int* begin() requires B { return &m; }
int* end() requires B { return &m + 1; }
BOOST_DESCRIBE_CLASS(X, (), (m), (), ());
};
int main() {
// {"m":1}
SHOW(boost::json::value_from(X<false>{.m=1}));
// [1]
SHOW(boost::json::value_from(X<true>{.m=1}));
} |
Since we require all members public for described classes, this basically leads us again to |
Thank you for the issue. I'll review the conversion category ranking. |
The more I think about this, the more it seems to me that when a type matches more than one of these categories, the result should be a compile error. Choosing one of them is rarely the best course of action, all things considered. |
So, if a type matches several categories, the user would have to define all but one to derive from |
I think instead of having N independent checks that are either true or false, you could have one variable template whose value is an enum that specifies the conversion kind. By default Then the user can just specialize the enum directly to do what they want. e.g. for the ip address example, which already demonstrates exactly the problem this issue is talking about... instead of specializing |
It's not really "instead of" because specializing an independent check means something different; it means "my type is detected as X, but it isn't X" (or vice versa). Whereas what we have here is that the individual checks are correct, the type is detected as X and Y because it's really X and Y, and we have to choose one for the representation. |
Right, the enum approach would be the answer to how you "choose one" for the representation. I'd rather affirmatively say "my type should be represented as an X" rather than say the negative "my type is not a Y" and continue to have nowhere in code that indicates that it's an X. |
std::array is why serialization via reflection should generally be opt-in and/or happen after other category checks; array was the first gotcha I ran into when doing unrestricted reflection. There's no real way to be ergonomic and not do category checks/requirements as things like variants have at least 3 ways to be represented in JSON that are common(submember like type, other member to determine type, the shape of the variant in serialized form(e.g. existence of members or just the basic JSON type string/bool/class/array). Enum's as int's has been my default without help because that is what we have in a pre-C++26 world(the underlying type). For reflection I added a mapping that defaults the to and from string conversions to the reflected names of the enumerators. But even simple things like numbers are not going to be easy for the user when half the numbers are encoded as strings for various reasons. There is loads of JSON in the wild like Regarding the optional, it fits into a bigger category of Nullable/Option types that are close but not the same. Specifically, constructing the result is different for things like shared_ptr/optional/raw pointers/... and I had to create a trait for this. I did prioritize it over other categories because that is how JSON is normally reflected here. Changing that is easy though as the categories can be specified by the user through the mapping and in 26 via attributes too. I have a trait for containers, but things mostly we can check that with begin/end/insert and that covered most cases; std::array stands out here. But this comes back to how to construct things is a common enough question that I had to make it a part of the api with some good defaults. |
As Andrzej pointed out recently in a draft paper on the Library Evolution reflector, the JSON conversion heuristics currently prioritize checking sequence-like before checking optional-like. This means that any optional-like types that are ranges will serialize like a sequence (
[42]
and[]
instead of42
andnull
) and fail to deserialize (due to lack ofpush_back
orinsert
), and, more importantly, an optional type that later adopts a range interface will break any users that use Boost.JSON (without even knowing if any such users exist).There are multiple other languages whose optional type is a sequence, so it's not far-fetched to suggest that a C++ one might exist also (regardless of whether
std::optional
ends up being one or not).There are at least two ways to preemptively address this issue that I can think of (though there are probably more solutions):
I think an
optional<pair<string, T>>
that is a range is also very nearly map-like, only rejected byemplace
probably not returning tuple-like. But it hypothetically could also (maybe you return the previous state as abool
?), so some care needs to be taken there too.I think a similar question might be asked of described classes that are also ranges. How should those be serialized, as objects or as arrays?
The text was updated successfully, but these errors were encountered: