William A. Kennington III | 14dd4eb | 2023-01-12 10:51:12 -0800 | [diff] [blame] | 1 | #include <fmt/format.h> |
| 2 | |
| 3 | #include <stdplus/net/addr/subnet.hpp> |
| 4 | |
William A. Kennington III | 160e382 | 2023-06-16 17:09:43 -0700 | [diff] [blame] | 5 | #include <string_view> |
| 6 | |
William A. Kennington III | 14dd4eb | 2023-01-12 10:51:12 -0800 | [diff] [blame] | 7 | #include <gtest/gtest.h> |
| 8 | |
William A. Kennington III | 160e382 | 2023-06-16 17:09:43 -0700 | [diff] [blame] | 9 | using std::literals::string_view_literals::operator""sv; |
| 10 | |
William A. Kennington III | 14dd4eb | 2023-01-12 10:51:12 -0800 | [diff] [blame] | 11 | namespace stdplus |
| 12 | { |
| 13 | |
| 14 | auto addr4Full = In4Addr{255, 255, 255, 255}; |
| 15 | auto addr6Full = In6Addr{255, 255, 255, 255, 255, 255, 255, 255, |
| 16 | 255, 255, 255, 255, 255, 255, 255, 255}; |
| 17 | |
| 18 | TEST(Subnet4, Basic) |
| 19 | { |
| 20 | EXPECT_NO_THROW(Subnet4(in_addr{0xffffffff}, 32)); |
| 21 | EXPECT_NO_THROW(Subnet4(addr4Full, 0)); |
| 22 | EXPECT_NO_THROW(Subnet4(in_addr{}, 10)); |
| 23 | EXPECT_THROW(Subnet4(in_addr{0xffffffff}, 33), std::invalid_argument); |
| 24 | EXPECT_THROW(Subnet4(in_addr{0xffffffff}, 64), std::invalid_argument); |
| 25 | |
| 26 | EXPECT_NE(Subnet4(in_addr{0xff}, 32), Subnet4(in_addr{}, 32)); |
| 27 | EXPECT_NE(Subnet4(in_addr{0xff}, 26), Subnet4(in_addr{0xff}, 27)); |
| 28 | EXPECT_EQ(Subnet4(in_addr{0xff}, 32), Subnet4(in_addr{0xff}, 32)); |
| 29 | EXPECT_EQ(Subnet4(in_addr{}, 1), Subnet4(in_addr{}, 1)); |
| 30 | } |
| 31 | |
| 32 | TEST(Subnet4, Network) |
| 33 | { |
| 34 | EXPECT_EQ((In4Addr{}), Subnet4(In4Addr{}, 32).network()); |
| 35 | EXPECT_EQ(addr4Full, Subnet4(addr4Full, 32).network()); |
| 36 | EXPECT_EQ((In4Addr{255, 255, 128, 0}), Subnet4(addr4Full, 17).network()); |
| 37 | EXPECT_EQ((In4Addr{255, 255, 0, 0}), Subnet4(addr4Full, 16).network()); |
| 38 | EXPECT_EQ((In4Addr{255, 254, 0, 0}), Subnet4(addr4Full, 15).network()); |
| 39 | EXPECT_EQ((In4Addr{}), Subnet4(addr4Full, 0).network()); |
| 40 | EXPECT_EQ((In4Addr{}), Subnet4(In4Addr{}, 0).network()); |
| 41 | } |
| 42 | |
| 43 | TEST(Subnet4, Contains) |
| 44 | { |
| 45 | EXPECT_TRUE(Subnet4(addr4Full, 32).contains(addr4Full)); |
| 46 | EXPECT_FALSE(Subnet4(addr4Full, 32).contains(In4Addr{255, 255, 255, 254})); |
| 47 | EXPECT_FALSE(Subnet4(addr4Full, 32).contains(In4Addr{})); |
| 48 | EXPECT_TRUE( |
| 49 | Subnet4(addr4Full, 17).contains(static_cast<in_addr>(addr4Full))); |
| 50 | EXPECT_TRUE(Subnet4(addr4Full, 17).contains(In4Addr{255, 255, 128, 134})); |
| 51 | EXPECT_FALSE(Subnet4(addr4Full, 17).contains(In4Addr{255, 255, 127, 132})); |
| 52 | EXPECT_TRUE(Subnet4(addr4Full, 14).contains(addr4Full)); |
| 53 | EXPECT_TRUE(Subnet4(addr4Full, 0).contains(addr4Full)); |
| 54 | EXPECT_TRUE(Subnet4(In4Addr{}, 0).contains(addr4Full)); |
| 55 | EXPECT_TRUE(Subnet4(addr4Full, 0).contains(In4Addr{})); |
| 56 | } |
| 57 | |
William A. Kennington III | 160e382 | 2023-06-16 17:09:43 -0700 | [diff] [blame] | 58 | TEST(Subnet4, FromStr) |
| 59 | { |
| 60 | constexpr FromStr<Subnet4> fs; |
| 61 | EXPECT_THROW(fs("10"sv), std::invalid_argument); |
| 62 | EXPECT_THROW(fs("/10"sv), std::invalid_argument); |
| 63 | EXPECT_THROW(fs("0.0.0.0"sv), std::invalid_argument); |
| 64 | EXPECT_THROW(fs("0.0.0.0/"sv), std::invalid_argument); |
| 65 | EXPECT_THROW(fs("::/80"sv), std::invalid_argument); |
| 66 | EXPECT_EQ((SubnetAny{in_addr{}, 30}), fs("0.0.0.0/30"sv)); |
William A. Kennington III | 39011c8 | 2023-06-28 17:49:30 -0700 | [diff] [blame] | 67 | EXPECT_EQ((SubnetAny{in_addr{}, 30}), "0.0.0.0/30"_sub4); |
William A. Kennington III | 160e382 | 2023-06-16 17:09:43 -0700 | [diff] [blame] | 68 | } |
| 69 | |
William A. Kennington III | d5957f5 | 2023-06-16 16:55:01 -0700 | [diff] [blame] | 70 | TEST(Subnet4, ToStr) |
| 71 | { |
| 72 | ToStrHandle<ToStr<Subnet4>> tsh; |
| 73 | EXPECT_EQ("0.0.0.0/16", tsh(Subnet4({}, 16))); |
| 74 | EXPECT_EQ("255.0.255.255/28", tsh(Subnet4(In4Addr{255, 0, 255, 255}, 28))); |
| 75 | EXPECT_EQ("a 1.2.3.4/32 b", |
| 76 | fmt::format("a {} b", Subnet4(In4Addr{1, 2, 3, 4}, 32))); |
| 77 | } |
| 78 | |
William A. Kennington III | 14dd4eb | 2023-01-12 10:51:12 -0800 | [diff] [blame] | 79 | TEST(Subnet6, Basic) |
| 80 | { |
| 81 | EXPECT_NO_THROW(Subnet6(in6_addr{0xff}, 128)); |
| 82 | EXPECT_NO_THROW(Subnet6(addr6Full, 0)); |
| 83 | EXPECT_NO_THROW(Subnet6(in6_addr{}, 65)); |
| 84 | EXPECT_THROW(Subnet6(in6_addr{0xff}, 129), std::invalid_argument); |
| 85 | EXPECT_THROW(Subnet6(in6_addr{0xff}, 150), std::invalid_argument); |
| 86 | |
| 87 | EXPECT_NE(Subnet6(in6_addr{0xff}, 32), Subnet6(in6_addr{}, 32)); |
| 88 | EXPECT_NE(Subnet6(in6_addr{0xff}, 26), Subnet6(in6_addr{0xff}, 27)); |
| 89 | EXPECT_EQ(Subnet6(in6_addr{0xff}, 32), Subnet6(in6_addr{0xff}, 32)); |
| 90 | EXPECT_EQ(Subnet6(in6_addr{}, 1), Subnet6(in6_addr{}, 1)); |
| 91 | } |
| 92 | |
| 93 | TEST(Subnet6, Network) |
| 94 | { |
| 95 | EXPECT_EQ(In6Addr(), Subnet6(In6Addr(), 128).network()); |
| 96 | EXPECT_EQ(addr6Full, Subnet6(addr6Full, 128).network()); |
| 97 | EXPECT_EQ((In6Addr{255, 255, 255, 255, 224}), |
| 98 | Subnet6(addr6Full, 35).network()); |
| 99 | EXPECT_EQ((In6Addr{255, 255, 255, 255}), Subnet6(addr6Full, 32).network()); |
| 100 | EXPECT_EQ((In6Addr{255, 255, 128, 0}), Subnet6(addr6Full, 17).network()); |
| 101 | EXPECT_EQ((In6Addr{255, 255, 0, 0}), Subnet6(addr6Full, 16).network()); |
| 102 | EXPECT_EQ((In6Addr{255, 254, 0, 0}), Subnet6(addr6Full, 15).network()); |
| 103 | EXPECT_EQ((In6Addr{}), Subnet6(addr6Full, 0).network()); |
| 104 | EXPECT_EQ((In6Addr{}), Subnet6(In6Addr{}, 0).network()); |
| 105 | } |
| 106 | |
| 107 | TEST(Subnet6, Contains) |
| 108 | { |
| 109 | auto addr6NFull = addr6Full; |
| 110 | addr6NFull.s6_addr[15] = 254; |
| 111 | EXPECT_TRUE(Subnet6(addr6Full, 128).contains(addr6Full)); |
| 112 | EXPECT_FALSE(Subnet6(addr6Full, 128).contains(addr6NFull)); |
| 113 | EXPECT_FALSE(Subnet6(addr6Full, 128).contains(In6Addr{})); |
| 114 | EXPECT_TRUE( |
| 115 | Subnet6(addr6Full, 127).contains(static_cast<in6_addr>(addr6Full))); |
| 116 | EXPECT_TRUE(Subnet6(addr6Full, 127).contains(addr6NFull)); |
| 117 | EXPECT_TRUE( |
| 118 | Subnet6(addr6Full, 33).contains(In6Addr{255, 255, 255, 255, 128, 255})); |
| 119 | EXPECT_FALSE( |
| 120 | Subnet6(addr6Full, 33).contains(In6Addr{255, 255, 255, 255, 127})); |
| 121 | EXPECT_TRUE(Subnet6(In6Addr{}, 33).contains(In6Addr{0, 0, 0, 0, 127})); |
| 122 | EXPECT_TRUE( |
| 123 | Subnet6(addr6Full, 14).contains(In6Addr{255, 255, 0, 0, 0, 0, 0, 145})); |
| 124 | EXPECT_FALSE(Subnet6(addr6Full, 14).contains(In6Addr{255, 127, 1})); |
| 125 | EXPECT_TRUE(Subnet6(addr6Full, 0).contains(addr6Full)); |
| 126 | EXPECT_TRUE(Subnet6(In6Addr{}, 0).contains(addr6Full)); |
| 127 | EXPECT_TRUE(Subnet6(addr6Full, 0).contains(In6Addr{})); |
| 128 | } |
| 129 | |
William A. Kennington III | 160e382 | 2023-06-16 17:09:43 -0700 | [diff] [blame] | 130 | TEST(Subnet6, FromStr) |
| 131 | { |
| 132 | constexpr FromStr<Subnet6> fs; |
| 133 | EXPECT_THROW(fs("10"sv), std::invalid_argument); |
| 134 | EXPECT_THROW(fs("/10"sv), std::invalid_argument); |
| 135 | EXPECT_THROW(fs("ff::"sv), std::invalid_argument); |
| 136 | EXPECT_THROW(fs("::/"sv), std::invalid_argument); |
| 137 | EXPECT_THROW(fs("0.0.0.0/0"sv), std::invalid_argument); |
| 138 | EXPECT_EQ((Subnet6{in6_addr{}, 80}), fs("::/80"sv)); |
William A. Kennington III | 39011c8 | 2023-06-28 17:49:30 -0700 | [diff] [blame] | 139 | EXPECT_EQ((Subnet6{in6_addr{}, 80}), "::/80"_sub6); |
William A. Kennington III | 160e382 | 2023-06-16 17:09:43 -0700 | [diff] [blame] | 140 | } |
| 141 | |
William A. Kennington III | d5957f5 | 2023-06-16 16:55:01 -0700 | [diff] [blame] | 142 | TEST(Subnet6, ToStr) |
| 143 | { |
| 144 | ToStrHandle<ToStr<Subnet6>> tsh; |
| 145 | EXPECT_EQ("::/0", tsh(Subnet6({}, 0))); |
| 146 | EXPECT_EQ("ff00::/128", tsh(Subnet6(In6Addr{0xff}, 128))); |
| 147 | EXPECT_EQ("a 102:304::/32 b", |
| 148 | fmt::format("a {} b", Subnet6(In6Addr{1, 2, 3, 4}, 32))); |
| 149 | } |
| 150 | |
William A. Kennington III | 14dd4eb | 2023-01-12 10:51:12 -0800 | [diff] [blame] | 151 | TEST(SubnetAny, Basic) |
| 152 | { |
| 153 | EXPECT_NO_THROW(SubnetAny(in_addr{0xffffffff}, 32)); |
| 154 | EXPECT_NO_THROW(SubnetAny(addr4Full, 0)); |
| 155 | EXPECT_NO_THROW(SubnetAny(InAnyAddr{addr4Full}, 0)); |
| 156 | EXPECT_NO_THROW(SubnetAny(in_addr{}, 10)); |
| 157 | EXPECT_THROW(SubnetAny(in_addr{0xffffffff}, 33), std::invalid_argument); |
| 158 | EXPECT_THROW(SubnetAny(InAnyAddr{in_addr{0xffffffff}}, 33), |
| 159 | std::invalid_argument); |
| 160 | EXPECT_THROW(SubnetAny(in_addr{0xffffffff}, 64), std::invalid_argument); |
| 161 | EXPECT_NO_THROW(SubnetAny(in6_addr{0xff}, 128)); |
| 162 | EXPECT_NO_THROW(SubnetAny(addr6Full, 0)); |
| 163 | EXPECT_NO_THROW(SubnetAny(InAnyAddr{addr6Full}, 0)); |
| 164 | EXPECT_NO_THROW(SubnetAny(in6_addr{}, 65)); |
| 165 | EXPECT_THROW(SubnetAny(in6_addr{0xff}, 129), std::invalid_argument); |
| 166 | EXPECT_THROW(SubnetAny(InAnyAddr{in6_addr{0xff}}, 129), |
| 167 | std::invalid_argument); |
| 168 | EXPECT_THROW(SubnetAny(in6_addr{0xff}, 150), std::invalid_argument); |
| 169 | |
| 170 | EXPECT_NO_THROW(SubnetAny(Subnet4(in_addr{}, 32))); |
| 171 | EXPECT_NO_THROW(SubnetAny(Subnet6(in6_addr{0xff}, 128))); |
| 172 | |
| 173 | EXPECT_NE(SubnetAny(in6_addr{0xff}, 32), Subnet6(in6_addr{}, 32)); |
| 174 | EXPECT_NE(Subnet6(in6_addr{0xff}, 26), SubnetAny(in6_addr{0xff}, 27)); |
| 175 | EXPECT_EQ(SubnetAny(in6_addr{0xff}, 32), Subnet6(in6_addr{0xff}, 32)); |
| 176 | EXPECT_EQ(SubnetAny(in6_addr{0xff}, 32), SubnetAny(in6_addr{0xff}, 32)); |
| 177 | EXPECT_NE(SubnetAny(in6_addr{0xff}, 32), Subnet4(in_addr{0xff}, 32)); |
| 178 | EXPECT_NE(SubnetAny(in_addr{0xff}, 32), Subnet4(in_addr{}, 32)); |
| 179 | EXPECT_NE(Subnet4(in_addr{0xff}, 26), SubnetAny(in_addr{0xff}, 27)); |
| 180 | EXPECT_EQ(SubnetAny(in_addr{0xff}, 32), Subnet4(in_addr{0xff}, 32)); |
| 181 | EXPECT_EQ(SubnetAny(in_addr{0xff}, 32), SubnetAny(in_addr{0xff}, 32)); |
| 182 | } |
| 183 | |
| 184 | TEST(SubnetAny, Network) |
| 185 | { |
| 186 | EXPECT_EQ(In6Addr(), SubnetAny(In6Addr(), 128).network()); |
| 187 | EXPECT_EQ(addr6Full, SubnetAny(addr6Full, 128).network()); |
| 188 | EXPECT_EQ(In6Addr(), SubnetAny(addr6Full, 0).network()); |
| 189 | EXPECT_EQ(In4Addr(), SubnetAny(In4Addr(), 32).network()); |
| 190 | EXPECT_EQ(addr4Full, SubnetAny(addr4Full, 32).network()); |
| 191 | EXPECT_EQ(In4Addr(), SubnetAny(addr4Full, 0).network()); |
| 192 | } |
| 193 | |
| 194 | TEST(SubnetAny, Contains) |
| 195 | { |
| 196 | EXPECT_TRUE(SubnetAny(addr6Full, 128).contains(addr6Full)); |
| 197 | EXPECT_TRUE(SubnetAny(addr6Full, 128).contains(InAnyAddr{addr6Full})); |
| 198 | EXPECT_FALSE(SubnetAny(addr6Full, 128).contains(in6_addr{})); |
| 199 | EXPECT_FALSE(SubnetAny(addr6Full, 128).contains(InAnyAddr(in6_addr{}))); |
| 200 | EXPECT_TRUE(SubnetAny(addr4Full, 32).contains(addr4Full)); |
| 201 | EXPECT_TRUE(SubnetAny(addr4Full, 32).contains(InAnyAddr{addr4Full})); |
| 202 | EXPECT_FALSE(SubnetAny(addr4Full, 32).contains(in_addr{})); |
| 203 | EXPECT_FALSE(SubnetAny(addr4Full, 32).contains(InAnyAddr{In4Addr{}})); |
| 204 | } |
| 205 | |
William A. Kennington III | 160e382 | 2023-06-16 17:09:43 -0700 | [diff] [blame] | 206 | TEST(SubnetAny, FromStr) |
| 207 | { |
| 208 | constexpr FromStr<SubnetAny> fs; |
| 209 | EXPECT_THROW(fs("10"sv), std::invalid_argument); |
| 210 | EXPECT_THROW(fs("/10"sv), std::invalid_argument); |
| 211 | EXPECT_THROW(fs("0.0.0.0"sv), std::invalid_argument); |
| 212 | EXPECT_THROW(fs("0.0.0.0/"sv), std::invalid_argument); |
| 213 | EXPECT_EQ((SubnetAny{in_addr{}, 0}), fs("0.0.0.0/0"sv)); |
| 214 | EXPECT_EQ((SubnetAny{in_addr{}, 30}), fs("0.0.0.0/30"sv)); |
William A. Kennington III | 39011c8 | 2023-06-28 17:49:30 -0700 | [diff] [blame] | 215 | EXPECT_EQ((SubnetAny{in_addr{}, 30}), "0.0.0.0/30"_sub); |
William A. Kennington III | 160e382 | 2023-06-16 17:09:43 -0700 | [diff] [blame] | 216 | EXPECT_EQ((SubnetAny{in6_addr{}, 80}), fs("::/80"sv)); |
William A. Kennington III | 39011c8 | 2023-06-28 17:49:30 -0700 | [diff] [blame] | 217 | EXPECT_EQ((SubnetAny{in6_addr{}, 80}), "::/80"_sub); |
William A. Kennington III | 160e382 | 2023-06-16 17:09:43 -0700 | [diff] [blame] | 218 | } |
| 219 | |
William A. Kennington III | d5957f5 | 2023-06-16 16:55:01 -0700 | [diff] [blame] | 220 | TEST(SubnetAny, ToStr) |
| 221 | { |
| 222 | ToStrHandle<ToStr<SubnetAny>> tsh; |
| 223 | EXPECT_EQ("0.0.0.0/16", tsh(SubnetAny(In4Addr{}, 16))); |
| 224 | EXPECT_EQ("ff00::/128", tsh(SubnetAny(In6Addr{0xff}, 128))); |
| 225 | EXPECT_EQ("a 102:304::/32 b", |
| 226 | fmt::format("a {} b", SubnetAny(In6Addr{1, 2, 3, 4}, 32))); |
| 227 | EXPECT_EQ("a 1.2.3.4/32 b", |
| 228 | fmt::format("a {} b", SubnetAny(In4Addr{1, 2, 3, 4}, 32))); |
| 229 | } |
| 230 | |
William A. Kennington III | c870ac3 | 2023-06-17 17:13:45 -0700 | [diff] [blame] | 231 | TEST(Ops, MaskToPfx) |
| 232 | { |
| 233 | EXPECT_EQ(32, maskToPfx(In4Addr{0xff, 0xff, 0xff, 0xff})); |
| 234 | EXPECT_EQ(28, maskToPfx(In4Addr{0xff, 0xff, 0xff, 0xf0})); |
| 235 | EXPECT_EQ(16, maskToPfx(In4Addr{0xff, 0xff, 0, 0})); |
| 236 | EXPECT_EQ(0, maskToPfx(In4Addr{})); |
| 237 | |
| 238 | EXPECT_THROW(maskToPfx(In4Addr{0, 0x8}), std::invalid_argument); |
| 239 | EXPECT_THROW(maskToPfx(In4Addr{0x40}), std::invalid_argument); |
| 240 | } |
| 241 | |
William A. Kennington III | 14dd4eb | 2023-01-12 10:51:12 -0800 | [diff] [blame] | 242 | } // namespace stdplus |