option4_addrlst_unittest.cc 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275
  1. // Copyright (C) 2011-2016 Internet Systems Consortium, Inc. ("ISC")
  2. //
  3. // This Source Code Form is subject to the terms of the Mozilla Public
  4. // License, v. 2.0. If a copy of the MPL was not distributed with this
  5. // file, You can obtain one at http://mozilla.org/MPL/2.0/.
  6. #include <config.h>
  7. #include <asiolink/io_address.h>
  8. #include <dhcp/dhcp4.h>
  9. #include <dhcp/option.h>
  10. #include <dhcp/option4_addrlst.h>
  11. #include <util/buffer.h>
  12. #include <gtest/gtest.h>
  13. #include <boost/scoped_ptr.hpp>
  14. #include <iostream>
  15. #include <sstream>
  16. #include <arpa/inet.h>
  17. using namespace std;
  18. using namespace isc;
  19. using namespace isc::dhcp;
  20. using namespace isc::asiolink;
  21. using namespace isc::util;
  22. using boost::scoped_ptr;
  23. namespace {
  24. // a sample data (list of 4 addresses)
  25. const uint8_t sampledata[] = {
  26. 192, 0, 2, 3, // 192.0.2.3
  27. 255, 255, 255, 0, // 255.255.255.0 - popular netmask
  28. 0, 0, 0 , 0, // used for default routes or (any address)
  29. 127, 0, 0, 1 // loopback
  30. };
  31. // expected on-wire format for an option with 1 address
  32. const uint8_t expected1[] = { // 1 address
  33. DHO_DOMAIN_NAME_SERVERS, 4, // type, length
  34. 192, 0, 2, 3, // 192.0.2.3
  35. };
  36. // expected on-wire format for an option with 4 addresses
  37. const uint8_t expected4[] = { // 4 addresses
  38. 254, 16, // type = 254, len = 16
  39. 192, 0, 2, 3, // 192.0.2.3
  40. 255, 255, 255, 0, // 255.255.255.0 - popular netmask
  41. 0, 0, 0 ,0, // used for default routes or (any address)
  42. 127, 0, 0, 1 // loopback
  43. };
  44. class Option4AddrLstTest : public ::testing::Test {
  45. protected:
  46. Option4AddrLstTest():
  47. vec_(vector<uint8_t>(300, 0)) // 300 bytes long filled with 0s
  48. {
  49. sampleAddrs_.push_back(IOAddress("192.0.2.3"));
  50. sampleAddrs_.push_back(IOAddress("255.255.255.0"));
  51. sampleAddrs_.push_back(IOAddress("0.0.0.0"));
  52. sampleAddrs_.push_back(IOAddress("127.0.0.1"));
  53. }
  54. vector<uint8_t> vec_;
  55. Option4AddrLst::AddressContainer sampleAddrs_;
  56. };
  57. TEST_F(Option4AddrLstTest, parse1) {
  58. memcpy(&vec_[0], sampledata, sizeof(sampledata));
  59. // Just one address
  60. scoped_ptr<Option4AddrLst> opt1;
  61. EXPECT_NO_THROW(
  62. opt1.reset(new Option4AddrLst(DHO_DOMAIN_NAME_SERVERS,
  63. vec_.begin(),
  64. vec_.begin()+4));
  65. // Use just first address (4 bytes), not the whole
  66. // sampledata
  67. );
  68. EXPECT_EQ(Option::V4, opt1->getUniverse());
  69. EXPECT_EQ(DHO_DOMAIN_NAME_SERVERS, opt1->getType());
  70. EXPECT_EQ(6, opt1->len()); // 2 (header) + 4 (1x IPv4 addr)
  71. Option4AddrLst::AddressContainer addrs = opt1->getAddresses();
  72. ASSERT_EQ(1, addrs.size());
  73. EXPECT_EQ("192.0.2.3", addrs[0].toText());
  74. EXPECT_NO_THROW(opt1.reset());
  75. }
  76. TEST_F(Option4AddrLstTest, parse4) {
  77. vector<uint8_t> buffer(300, 0); // 300 bytes long filled with 0s
  78. memcpy(&buffer[0], sampledata, sizeof(sampledata));
  79. // 4 addresses
  80. scoped_ptr<Option4AddrLst> opt4;
  81. EXPECT_NO_THROW(
  82. opt4.reset(new Option4AddrLst(254,
  83. buffer.begin(),
  84. buffer.begin()+sizeof(sampledata)));
  85. );
  86. EXPECT_EQ(Option::V4, opt4->getUniverse());
  87. EXPECT_EQ(254, opt4->getType());
  88. EXPECT_EQ(18, opt4->len()); // 2 (header) + 16 (4x IPv4 addrs)
  89. Option4AddrLst::AddressContainer addrs = opt4->getAddresses();
  90. ASSERT_EQ(4, addrs.size());
  91. EXPECT_EQ("192.0.2.3", addrs[0].toText());
  92. EXPECT_EQ("255.255.255.0", addrs[1].toText());
  93. EXPECT_EQ("0.0.0.0", addrs[2].toText());
  94. EXPECT_EQ("127.0.0.1", addrs[3].toText());
  95. EXPECT_NO_THROW(opt4.reset());
  96. }
  97. TEST_F(Option4AddrLstTest, assembly1) {
  98. scoped_ptr<Option4AddrLst> opt;
  99. EXPECT_NO_THROW(
  100. opt.reset(new Option4AddrLst(DHO_DOMAIN_NAME_SERVERS,
  101. IOAddress("192.0.2.3")));
  102. );
  103. EXPECT_EQ(Option::V4, opt->getUniverse());
  104. EXPECT_EQ(DHO_DOMAIN_NAME_SERVERS, opt->getType());
  105. Option4AddrLst::AddressContainer addrs = opt->getAddresses();
  106. ASSERT_EQ(1, addrs.size() );
  107. EXPECT_EQ("192.0.2.3", addrs[0].toText());
  108. OutputBuffer buf(100);
  109. EXPECT_NO_THROW(
  110. opt->pack(buf);
  111. );
  112. ASSERT_EQ(6, opt->len());
  113. ASSERT_EQ(6, buf.getLength());
  114. EXPECT_EQ(0, memcmp(expected1, buf.getData(), 6));
  115. EXPECT_NO_THROW(opt.reset());
  116. // This is old-fashioned option. We don't serve IPv6 types here!
  117. EXPECT_THROW(
  118. opt.reset(new Option4AddrLst(DHO_DOMAIN_NAME_SERVERS,
  119. IOAddress("2001:db8::1"))),
  120. BadValue
  121. );
  122. }
  123. TEST_F(Option4AddrLstTest, assembly4) {
  124. scoped_ptr<Option4AddrLst> opt;
  125. EXPECT_NO_THROW(
  126. opt.reset(new Option4AddrLst(254, sampleAddrs_));
  127. );
  128. EXPECT_EQ(Option::V4, opt->getUniverse());
  129. EXPECT_EQ(254, opt->getType());
  130. Option4AddrLst::AddressContainer addrs = opt->getAddresses();
  131. ASSERT_EQ(4, addrs.size() );
  132. EXPECT_EQ("192.0.2.3", addrs[0].toText());
  133. EXPECT_EQ("255.255.255.0", addrs[1].toText());
  134. EXPECT_EQ("0.0.0.0", addrs[2].toText());
  135. EXPECT_EQ("127.0.0.1", addrs[3].toText());
  136. OutputBuffer buf(100);
  137. EXPECT_NO_THROW(
  138. opt->pack(buf);
  139. );
  140. ASSERT_EQ(18, opt->len()); // 2(header) + 4xsizeof(IPv4addr)
  141. ASSERT_EQ(18, buf.getLength());
  142. ASSERT_EQ(0, memcmp(expected4, buf.getData(), 18));
  143. EXPECT_NO_THROW(opt.reset());
  144. // This is old-fashioned option. We don't serve IPv6 types here!
  145. sampleAddrs_.push_back(IOAddress("2001:db8::1"));
  146. EXPECT_THROW(
  147. opt.reset(new Option4AddrLst(DHO_DOMAIN_NAME_SERVERS, sampleAddrs_)),
  148. BadValue
  149. );
  150. }
  151. // This test verifies that an option (e.g., mobile-ip-home-agent) can be empty.
  152. TEST_F(Option4AddrLstTest, empty) {
  153. scoped_ptr<Option4AddrLst> opt;
  154. // the mobile-ip-home-agent option can be empty
  155. EXPECT_NO_THROW(opt.reset(new Option4AddrLst(DHO_HOME_AGENT_ADDRS)));
  156. Option4AddrLst::AddressContainer addrs = opt->getAddresses();
  157. ASSERT_EQ(0, addrs.size());
  158. EXPECT_NO_THROW(opt.reset());
  159. }
  160. TEST_F(Option4AddrLstTest, setAddress) {
  161. scoped_ptr<Option4AddrLst> opt;
  162. EXPECT_NO_THROW(
  163. opt.reset(new Option4AddrLst(123, IOAddress("1.2.3.4")));
  164. );
  165. opt->setAddress(IOAddress("192.0.255.255"));
  166. Option4AddrLst::AddressContainer addrs = opt->getAddresses();
  167. ASSERT_EQ(1, addrs.size() );
  168. EXPECT_EQ("192.0.255.255", addrs[0].toText());
  169. // We should accept IPv4-only addresses.
  170. EXPECT_THROW(
  171. opt->setAddress(IOAddress("2001:db8::1")),
  172. BadValue
  173. );
  174. EXPECT_NO_THROW(opt.reset());
  175. }
  176. TEST_F(Option4AddrLstTest, setAddresses) {
  177. scoped_ptr<Option4AddrLst> opt;
  178. EXPECT_NO_THROW(
  179. opt.reset(new Option4AddrLst(123)); // Empty list
  180. );
  181. opt->setAddresses(sampleAddrs_);
  182. Option4AddrLst::AddressContainer addrs = opt->getAddresses();
  183. ASSERT_EQ(4, addrs.size() );
  184. EXPECT_EQ("192.0.2.3", addrs[0].toText());
  185. EXPECT_EQ("255.255.255.0", addrs[1].toText());
  186. EXPECT_EQ("0.0.0.0", addrs[2].toText());
  187. EXPECT_EQ("127.0.0.1", addrs[3].toText());
  188. // We should accept IPv4-only addresses.
  189. sampleAddrs_.push_back(IOAddress("2001:db8::1"));
  190. EXPECT_THROW(
  191. opt->setAddresses(sampleAddrs_),
  192. BadValue
  193. );
  194. EXPECT_NO_THROW(opt.reset());
  195. }
  196. // This test checks that the option holding IPv4 address list can
  197. // be converted to textual format.
  198. TEST_F(Option4AddrLstTest, toText) {
  199. Option4AddrLst opt(111);
  200. // Generate a few IPv4 addresses.
  201. Option4AddrLst::AddressContainer addresses;
  202. for (int i = 2; i < 6; ++i) {
  203. std::stringstream s;
  204. s << "192.0.2." << i;
  205. addresses.push_back(IOAddress(s.str()));
  206. }
  207. opt.setAddresses(addresses);
  208. EXPECT_EQ("type=111, len=016: 192.0.2.2 192.0.2.3 192.0.2.4 192.0.2.5",
  209. opt.toText());
  210. }
  211. } // namespace