-
Notifications
You must be signed in to change notification settings - Fork 142
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
Allow per-element comparison of string against non-string iterable #242
base: develop
Are you sure you want to change the base?
Conversation
This is being superseded by the branch |
2648a1b
to
5082617
Compare
@raffienficiaud thanks, I've tested and I still need the change for comparisons to work. I've rebased my patch and force pushed. My commit is the last one, 5082617 |
&& !(unit_test::is_cstring_comparable<Lhs>::value \ | ||
&& unit_test::is_cstring_comparable<Rhs>::value)>::type> { \ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This will likely break other stuff: the collection comparison is mutually exclusive with the cstring comparison operator definition in include/boost/test/tools/cstring_comparison_op.hpp
. The latter calls the collection comparison, but it should be the cstring comparison that captures the expression.
If the condition changes in this collection comparison (include/boost/test/tools/collection_comparison_op.hpp
), the same changes should be reflected in the cstring comparison set of operators.
The only way I see is
- change in
include/boost/test/tools/cstring_comparison_op.hpp
: making theenable_if
with an||
condition:
#define DEFINE_CSTRING_COMPARISON( oper, name, rev, name_inverse ) \
template<typename Lhs,typename Rhs>
struct name<Lhs,Rhs,typename boost::enable_if_c<
( unit_test::is_cstring_comparable<Lhs>::value
|| unit_test::is_cstring_comparable<Rhs>::value)
>::type >
- change in
include/boost/test/tools/collection_comparison_op.hpp
to have!(unit_test::is_cstring_comparable<Lhs>::value || unit_test::is_cstring_comparable<Rhs>::value)>::type
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Well, both sides need to be cstring comparable for cstring comparison to apply, so the operator should be &&
- which is what I'm proposing.
Currently, if both sides are cstring comparable then the cstring comparison is selected; if neither are cstring comparable the collection comparison is selected; if only one side is cstring comparable then neither is selected and compilation fails.
I'm proposing to fill in the gap so that if only one side is cstring comparable it falls back to collection comparison. If it'd be clearer, I could de Morgan's rewrite the collection comparison condition to have (!unit_test::is_cstring_comparable<Lhs>::value || !unit_test::is_cstring_comparable<Rhs>::value)
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If any side is cstring
it should be captured by the cstring
operation, the other side should be "something" that has the same forward semantics as the cstring
, and that can be transformed via the deduce_cstring_transform
.
Currently the string comparison operator is forwarding to the collection comparison, but some string specific operations may happen there (such as encoding-aware comparison) prior to sending to the collection comparison.
There should be a switch that uses deduce_cstring_transform
for cstring
or a no-op otherwise, for the rhs_char_type
part. It should however be restricted enough to avoid capturing:
BOOST_TEST( "abc" == std::vector<int>{'a', 'b', 'c'}).
For that, I guess an extra condition to the unit_test::is_forward_iterable<Lhs>::value
should be that typename Rhs::value_type
is char
or wchar_t
(or using is_cstring_impl
via a pointer to value_type
, also Rhs
can be an array ...).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
One more thing, having the per_element
modifier should branch to
typename boost::enable_if_c<
(unit_test::is_cstring<Lhs>::value || unit_test::is_cstring<Rhs>::value),
assertion_result>::type
element_compare( Lhs const& lhs, Rhs const& rhs )
Isn't it working like that already?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
No, currently it (comparing e.g. std::string
to std::vector<char>
with per_element) isn't picked up by either DEFINE_COLLECTION_COMPARISON or DEFINE_CSTRING_COMPARISON so it falls through to DEFINE_CONST_OPER.
I'll have a look what it would take for DEFINE_CSTRING_COMPARISON to support non-string iterables on one side.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think there's any need for deduce_cstring_transform
to transform non-string iterables; does it really make sense to treat e.g. std::list<char>
as a string? As a user I'd be happy to use per_element()
, or if I really intended to compare as a string then boost::copy_range<std::string>
.
Leaving the forward iterable side as-is makes for a simpler implementation (and diff - see latest force push). Thanks!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
note - the reason to check that the other side is forward iterable is that this is required to access value_type
:
typedef name< \
typename lhs_char_type::value_type, \
typename rhs_char_type::value_type> elem_op; \
Without this restriction, the class_properties
tests fail.
@ecatmur thank you for the quick feedback. I commented on the PR, I can give a spin on my CI, travis takes ages. |
ad1dd5a
to
c51a94d
Compare
c51a94d
to
db191af
Compare
Codecov Report
Additional details and impacted files@@ Coverage Diff @@
## develop #242 +/- ##
==========================================
Coverage ? 55.77%
==========================================
Files ? 112
Lines ? 5897
Branches ? 2382
==========================================
Hits ? 3289
Misses ? 778
Partials ? 1830 ☔ View full report in Codecov by Sentry. |
Chiming in here-- I'm sad to report that @ecatmur has passed away over the new-years holiday period. Commemoration / memorial / donate to good cause page in case interested. This patch while in place at my organization, seemingly breaks compilation of some tests in Boost.MySQL as in Boost 1.85.0. Presumably due to some dependence on the new Boost.Charconv. To be perfectly honest with you, I didn't bother figuring out the compile error, I tested against our tests internally and the heavy hitters where I would have expected a problem all compiled and passed / expected failed in the right places; so internally we're just going to undo the patch and if someone was relying on it tell them to convert their container to a string or similar course of action; it's not as if such unit tests are performance critical and can't afford the copy or anything. Felt necessary to comment in case this would have gotten merged unknowingly. |
(of char).
Test.