portconfig_unittest.cc 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. // Copyright (C) 2011 Internet Systems Consortium, Inc. ("ISC")
  2. //
  3. // Permission to use, copy, modify, and/or distribute this software for any
  4. // purpose with or without fee is hereby granted, provided that the above
  5. // copyright notice and this permission notice appear in all copies.
  6. //
  7. // THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
  8. // REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
  9. // AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
  10. // INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
  11. // LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
  12. // OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  13. // PERFORMANCE OF THIS SOFTWARE.
  14. #include <server_common/portconfig.h>
  15. #include <cc/data.h>
  16. #include <exceptions/exceptions.h>
  17. #include <asiolink/asiolink.h>
  18. #include <asiodns/asiodns.h>
  19. #include <gtest/gtest.h>
  20. #include <string>
  21. using namespace isc::server_common::portconfig;
  22. using namespace isc::data;
  23. using namespace isc;
  24. using namespace std;
  25. using namespace isc::asiolink;
  26. using namespace isc::asiodns;
  27. namespace {
  28. /// Testcase for parseAddresses call (struct, nobody cares about private here)
  29. struct ParseAddresses : public ::testing::Test {
  30. AddressList result_;
  31. void empty(ElementPtr config, const string& name) {
  32. SCOPED_TRACE(name);
  33. EXPECT_NO_THROW(result_ = parseAddresses(config, "test"));
  34. EXPECT_TRUE(result_.empty());
  35. }
  36. template<class Exception>
  37. void invalidTest(const string& json, const string& name) {
  38. SCOPED_TRACE(name);
  39. ElementPtr config(Element::fromJSON(json));
  40. EXPECT_THROW(parseAddresses(config, "test"), Exception) <<
  41. "Should throw " << typeid(Exception).name();
  42. }
  43. };
  44. // Parse valid IPv4 address
  45. TEST_F(ParseAddresses, ipv4) {
  46. ElementPtr config(Element::fromJSON("["
  47. " {"
  48. " \"address\": \"192.0.2.1\","
  49. " \"port\": 53"
  50. " }"
  51. "]"));
  52. EXPECT_NO_THROW(result_ = parseAddresses(config, "test"));
  53. ASSERT_EQ(1, result_.size());
  54. EXPECT_EQ("192.0.2.1", result_[0].first);
  55. EXPECT_EQ(53, result_[0].second);
  56. }
  57. // Parse valid IPv6 address
  58. TEST_F(ParseAddresses, ipv6) {
  59. ElementPtr config(Element::fromJSON("["
  60. " {"
  61. " \"address\": \"2001:db8::1\","
  62. " \"port\": 53"
  63. " }"
  64. "]"));
  65. EXPECT_NO_THROW(result_ = parseAddresses(config, "test"));
  66. ASSERT_EQ(1, result_.size());
  67. EXPECT_EQ("2001:db8::1", result_[0].first);
  68. EXPECT_EQ(53, result_[0].second);
  69. }
  70. // Parse multiple addresses at once
  71. // (even the ports are different to see they are not mistaken)
  72. TEST_F(ParseAddresses, multi) {
  73. ElementPtr config(Element::fromJSON("["
  74. " {"
  75. " \"address\": \"2001:db8::1\","
  76. " \"port\": 53"
  77. " },"
  78. " {"
  79. " \"address\": \"192.0.2.1\","
  80. " \"port\": 54"
  81. " }"
  82. "]"));
  83. EXPECT_NO_THROW(result_ = parseAddresses(config, "test"));
  84. ASSERT_EQ(2, result_.size());
  85. EXPECT_EQ("2001:db8::1", result_[0].first);
  86. EXPECT_EQ(53, result_[0].second);
  87. EXPECT_EQ("192.0.2.1", result_[1].first);
  88. EXPECT_EQ(54, result_[1].second);
  89. }
  90. // Parse various versions of empty list
  91. TEST_F(ParseAddresses, empty) {
  92. empty(Element::fromJSON("[]"), "Empty list");
  93. empty(ElementPtr(new NullElement), "Null element");
  94. empty(ElementPtr(), "Null pointer");
  95. }
  96. // Reject invalid configs
  97. TEST_F(ParseAddresses, invalid) {
  98. invalidTest<TypeError>("{}", "Not a list");
  99. invalidTest<BadValue>("[{}]", "Empty element");
  100. invalidTest<TypeError>("[{"
  101. " \"port\": 1.5,"
  102. " \"address\": \"192.0.2.1\""
  103. "}]", "Float port");
  104. invalidTest<BadValue>("[{"
  105. " \"port\": -5,"
  106. " \"address\": \"192.0.2.1\""
  107. "}]", "Negative port");
  108. invalidTest<BadValue>("[{"
  109. " \"port\": 1000000,"
  110. " \"address\": \"192.0.2.1\""
  111. "}]", "Port too big");
  112. invalidTest<IOError>("[{"
  113. " \"port\": 53,"
  114. " \"address\": \"bad_address\""
  115. "}]", "Bad address");
  116. }
  117. // Test fixture for installListenAddresses
  118. struct InstallListenAddresses : public ::testing::Test {
  119. InstallListenAddresses() :
  120. dnss_(ios_, NULL, NULL, NULL)
  121. {
  122. valid_.push_back(AddressPair("127.0.0.1", 5288));
  123. valid_.push_back(AddressPair("::1", 5288));
  124. invalid_.push_back(AddressPair("192.0.2.2", 1));
  125. }
  126. IOService ios_;
  127. DNSService dnss_;
  128. AddressList store_;
  129. // We should be able to bind to these addresses
  130. AddressList valid_;
  131. // But this shouldn't work
  132. AddressList invalid_;
  133. // Check that the store_ addresses are the same as expected
  134. void checkAddresses(const AddressList& expected, const string& name) {
  135. SCOPED_TRACE(name);
  136. ASSERT_EQ(expected.size(), store_.size()) <<
  137. "Different amount of elements, not checking content";
  138. // Run in parallel trough the vectors
  139. for (AddressList::const_iterator ei(expected.begin()),
  140. si(store_.begin()); ei != expected.end(); ++ei, ++si) {
  141. EXPECT_EQ(ei->first, si->first);
  142. EXPECT_EQ(ei->second, si->second);
  143. }
  144. }
  145. };
  146. // Try switching valid addresses
  147. TEST_F(InstallListenAddresses, valid) {
  148. // First, bind to the valid addresses
  149. EXPECT_NO_THROW(installListenAddresses(valid_, store_, dnss_));
  150. checkAddresses(valid_, "Valid addresses");
  151. // TODO Maybe some test to actually connect to them
  152. // Try setting it back to nothing
  153. EXPECT_NO_THROW(installListenAddresses(AddressList(), store_, dnss_));
  154. checkAddresses(AddressList(), "No addresses");
  155. // Try switching back again
  156. EXPECT_NO_THROW(installListenAddresses(valid_, store_, dnss_));
  157. checkAddresses(valid_, "Valid addresses");
  158. }
  159. // Try if rollback works
  160. TEST_F(InstallListenAddresses, rollback) {
  161. // Set some addresses
  162. EXPECT_NO_THROW(installListenAddresses(valid_, store_, dnss_));
  163. checkAddresses(valid_, "Before rollback");
  164. // This should not bind them, but should leave the original addresses
  165. EXPECT_THROW(installListenAddresses(invalid_, store_, dnss_), IOError);
  166. checkAddresses(valid_, "After rollback");
  167. }
  168. }