This patch improves the cnetaddr link-local test in net_tests
by making it platform-specific and by using a higher scoped id value to alleviate recent CI issues.
Closes #21682.
The intermittently failing BOOST_CHECK
is:
I don’t understand why we test if ToString()
output includes %zone_index
: it clearly doesn’t on some platforms, so we cannot rely on it anyways. Then why test it? :)
My understanding when working on this was that Apple might be going in its own direction with this starting with macOS 10.14 but it still seemed worthwhile to test our code on platforms like Linux and Windows where we can.
But the test is not conditional on platform, so AFAICT we’re testing that ToString
either contains %zone_id
(the addr_str == scoped_addr
case) or that it doesn’t contain %zone_id
(the addr_str == "fe80:0:0:0:0:0:0:1"
case)? :)
In other words: we’re effectively not testing anything meaningful in a test that is intermittently failing, no? :)
If so, why not simply remove it? :)
If you test only the fallback, I found that the change fails on all CIs except the macOS 10.14 one:
0- BOOST_CHECK(addr_str == scoped_addr || addr_str == "fe80:0:0:0:0:0:0:1");
1+ BOOST_CHECK(addr_str == "fe80:0:0:0:0:0:0:1");
$ src/test/test_bitcoin -t net_tests Running 14 test cases… test/net_tests.cpp(311): error: in “net_tests/cnetaddr_basic”: check addr_str == “fe80:0:0:0:0:0:0:1” has failed
*** 1 failure is detected in the test module “Bitcoin Core Test Suite”
I didn’t get if you agree with my statement that the intermittently failing BOOST_CHECK
is basically doing:
0BOOST_CHECK(to_string_does_contain_zone_id || to_string_does_not_contain_zone_id);
And if so, is there any point in keeping it around? :)
FWIW, patch against current master
to show the issue more clearly:
0diff --git a/src/test/net_tests.cpp b/src/test/net_tests.cpp
1index 8eab26f3d..e7c3475ef 100644
2--- a/src/test/net_tests.cpp
3+++ b/src/test/net_tests.cpp
4@@ -308,8 +308,9 @@ BOOST_AUTO_TEST_CASE(cnetaddr_basic)
5 BOOST_REQUIRE(addr.IsIPv6());
6 BOOST_CHECK(!addr.IsBindAny());
7 const std::string addr_str{addr.ToString()};
8- BOOST_CHECK(addr_str == scoped_addr || addr_str == "fe80:0:0:0:0:0:0:1");
9- // The fallback case "fe80:0:0:0:0:0:0:1" is needed for macOS 10.14/10.15 and (probably) later.
10+ const bool to_string_does_contain_zone_id = addr_str == scoped_addr;
11+ const bool to_string_does_not_contain_zone_id = addr_str == "fe80:0:0:0:0:0:0:1";
12+ BOOST_CHECK(to_string_does_contain_zone_id || to_string_does_not_contain_zone_id);
13 // Test that the delimiter "%" and default zone id of 0 can be omitted for the default scope.
14 BOOST_REQUIRE(LookupHost(link_local + "%0", addr, false));
15 BOOST_REQUIRE(addr.IsValid());
It looks like I should document the test better to communicate the context and information. I’ll propose that separately.
Why can’t that be done as part of this change?
Ok. The CI has been green, and #21696 showed that macOS remains the only CI that fails the scoped_addr
case.
Let’s try (a) improving the documentation and (b) testing scoped_addr
by platform.
Edit: will update to BOOST_CHECK_EQUAL
if this works.
BOOST_CHECK( == )
replaced by BOOST_CHECK_EQUAL( , )
for a 4th CI run. I hope this addresses the feedback above while maintaining coverage and adding clarity and context.
The following sections might be updated with supplementary metadata relevant to reviewers and maintainers.
No conflicts as of last run.
299@@ -300,16 +300,21 @@ BOOST_AUTO_TEST_CASE(cnetaddr_basic)
300
301 // IPv6, scoped/link-local. See https://tools.ietf.org/html/rfc4007
302 // We support non-negative decimal integers (uint32_t) as zone id indices.
303- // Test with a fairly-high value, e.g. 32, to avoid locally reserved ids.
304+ // Test with a high value, e.g. 4096, to avoid any local ids in use.
305 const std::string link_local{"fe80::1"};
306- const std::string scoped_addr{link_local + "%32"};
307+ const std::string scoped_addr{link_local + "%4096"};
Good question. From what I understand, the scope ids / zone indices depend on the number of local network interfaces when there are multiple on a single link-local address. Whereas many machines may have the first 8 or 16 decimal integer scope ids in use, the previous test value of 32 seems to be on the high end of the range. I’m not aware of one that is guaranteed to be free, but a value in the thousands (or tens of thousands) might be reasonably worth trying.
Resources:
If so, why not simply remove it? :)
I think it is better to improve the test and contextual information it provides and I spent time in a way that I thought was constructive and helpful re-verifying and working to do that. Unfortunately, it’s an order of magnitude more work to refute statements like “not very meaningful” in the #21689 PR title and the comment above, than it is to say them and just erase coverage and (I think) valuable context.
@practicalswift does this now address your feedback?
The BOOST_CHECK(to_string_does_contain_zone_id || to_string_does_not_contain_zone_id)
issue has been addressed.
I think the issue @MarcoFalke raised about the test outcome being dependent on the user’s local configuration remains unfortunately.
314+ // Expect macOS to return the address in this format, without zone ids.
315+ BOOST_CHECK_EQUAL(addr_str, "fe80:0:0:0:0:0:0:1");
316+#else
317+ // Expect normal scoped address functionality on the other platforms.
318+ BOOST_CHECK_EQUAL(addr_str, scoped_addr);
319+#endif
0 BOOST_CHECK(addr.ToString().starts_with("fe80:0:0:0:0:0:0:1"));
Maybe just check the prefix if the suffix is unstable?