|
@@ -0,0 +1,378 @@
|
|
|
+// Copyright (C) 2014 Internet Systems Consortium, Inc. ("ISC")
|
|
|
+//
|
|
|
+// Permission to use, copy, modify, and/or distribute this software for any
|
|
|
+// purpose with or without fee is hereby granted, provided that the above
|
|
|
+// copyright notice and this permission notice appear in all copies.
|
|
|
+//
|
|
|
+// THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
|
|
|
+// REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
|
+// AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
|
+// INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
|
|
+// LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
|
|
|
+// OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
|
+// PERFORMANCE OF THIS SOFTWARE.
|
|
|
+
|
|
|
+#include <config.h>
|
|
|
+#include <asiolink/io_address.h>
|
|
|
+#include <cc/data.h>
|
|
|
+#include <dhcp/dhcp4.h>
|
|
|
+#include <dhcp/tests/iface_mgr_test_config.h>
|
|
|
+#include <dhcp4/tests/dhcp4_test_utils.h>
|
|
|
+#include <dhcp4/tests/dhcp4_client.h>
|
|
|
+
|
|
|
+using namespace isc;
|
|
|
+using namespace isc::asiolink;
|
|
|
+using namespace isc::data;
|
|
|
+using namespace isc::dhcp;
|
|
|
+using namespace isc::dhcp::test;
|
|
|
+
|
|
|
+namespace {
|
|
|
+
|
|
|
+/// @brief Set of JSON configurations used throughout the Rebind tests.
|
|
|
+///
|
|
|
+/// - Configuration 0:
|
|
|
+/// - Used for testing direct traffic
|
|
|
+/// - 1 subnet: 10.0.0.0/24
|
|
|
+/// - 1 pool: 10.0.0.10-10.0.0.100
|
|
|
+/// - Router option present: 10.0.0.200 and 10.0.0.201
|
|
|
+/// - Domain Name Server option present: 10.0.0.202, 10.0.0.203.
|
|
|
+/// - Log Servers option present: 192.0.2.200 and 192.0.2.201
|
|
|
+/// - Quotes Servers option present: 192.0.2.202, 192.0.2.203.
|
|
|
+///
|
|
|
+/// - Configuration 1:
|
|
|
+/// - Use for testing relayed messages
|
|
|
+/// - 1 subnet: 192.0.2.0/24
|
|
|
+/// - Router option present: 192.0.2.200 and 192.0.2.201
|
|
|
+/// - Domain Name Server option present: 192.0.2.202, 192.0.2.203.
|
|
|
+/// - Log Servers option present: 192.0.2.200 and 192.0.2.201
|
|
|
+/// - Quotes Servers option present: 192.0.2.202, 192.0.2.203.
|
|
|
+const char* INFORM_CONFIGS[] = {
|
|
|
+// Configuration 0
|
|
|
+ "{ \"interfaces\": [ \"all\" ],"
|
|
|
+ "\"valid-lifetime\": 600,"
|
|
|
+ "\"subnet4\": [ { "
|
|
|
+ " \"subnet\": \"10.0.0.0/24\", "
|
|
|
+ " \"pool\": [ \"10.0.0.10-10.0.0.100\" ],"
|
|
|
+ " \"option-data\": [ {"
|
|
|
+ " \"name\": \"routers\","
|
|
|
+ " \"code\": 3,"
|
|
|
+ " \"data\": \"10.0.0.200,10.0.0.201\","
|
|
|
+ " \"csv-format\": true,"
|
|
|
+ " \"space\": \"dhcp4\""
|
|
|
+ " },"
|
|
|
+ " {"
|
|
|
+ " \"name\": \"domain-name-servers\","
|
|
|
+ " \"code\": 6,"
|
|
|
+ " \"data\": \"10.0.0.202,10.0.0.203\","
|
|
|
+ " \"csv-format\": true,"
|
|
|
+ " \"space\": \"dhcp4\""
|
|
|
+ " },"
|
|
|
+ " {"
|
|
|
+ " \"name\": \"log-servers\","
|
|
|
+ " \"code\": 7,"
|
|
|
+ " \"data\": \"10.0.0.200,10.0.0.201\","
|
|
|
+ " \"csv-format\": true,"
|
|
|
+ " \"space\": \"dhcp4\""
|
|
|
+ " },"
|
|
|
+ " {"
|
|
|
+ " \"name\": \"cookie-servers\","
|
|
|
+ " \"code\": 8,"
|
|
|
+ " \"data\": \"10.0.0.202,10.0.0.203\","
|
|
|
+ " \"csv-format\": true,"
|
|
|
+ " \"space\": \"dhcp4\""
|
|
|
+ " } ]"
|
|
|
+ " } ]"
|
|
|
+ "}",
|
|
|
+
|
|
|
+// Configuration 1
|
|
|
+ "{ \"interfaces\": [ \"all\" ],"
|
|
|
+ "\"valid-lifetime\": 600,"
|
|
|
+ "\"subnet4\": [ { "
|
|
|
+ " \"subnet\": \"192.0.2.0/24\", "
|
|
|
+ " \"option-data\": [ {"
|
|
|
+ " \"name\": \"routers\","
|
|
|
+ " \"code\": 3,"
|
|
|
+ " \"data\": \"192.0.2.200,192.0.2.201\","
|
|
|
+ " \"csv-format\": true,"
|
|
|
+ " \"space\": \"dhcp4\""
|
|
|
+ " },"
|
|
|
+ " {"
|
|
|
+ " \"name\": \"domain-name-servers\","
|
|
|
+ " \"code\": 6,"
|
|
|
+ " \"data\": \"192.0.2.202,192.0.2.203\","
|
|
|
+ " \"csv-format\": true,"
|
|
|
+ " \"space\": \"dhcp4\""
|
|
|
+ " },"
|
|
|
+ " {"
|
|
|
+ " \"name\": \"log-servers\","
|
|
|
+ " \"code\": 7,"
|
|
|
+ " \"data\": \"10.0.0.200,10.0.0.201\","
|
|
|
+ " \"csv-format\": true,"
|
|
|
+ " \"space\": \"dhcp4\""
|
|
|
+ " },"
|
|
|
+ " {"
|
|
|
+ " \"name\": \"cookie-servers\","
|
|
|
+ " \"code\": 8,"
|
|
|
+ " \"data\": \"10.0.0.202,10.0.0.203\","
|
|
|
+ " \"csv-format\": true,"
|
|
|
+ " \"space\": \"dhcp4\""
|
|
|
+ " } ]"
|
|
|
+ " } ]"
|
|
|
+ "}"
|
|
|
+};
|
|
|
+
|
|
|
+/// @brief Test fixture class for testing DHCPINFORM.
|
|
|
+class InformTest : public Dhcpv4SrvTest {
|
|
|
+public:
|
|
|
+
|
|
|
+ /// @brief Constructor.
|
|
|
+ ///
|
|
|
+ /// Sets up fake interfaces.
|
|
|
+ InformTest()
|
|
|
+ : Dhcpv4SrvTest(),
|
|
|
+ iface_mgr_test_config_(true) {
|
|
|
+ IfaceMgr::instance().openSockets4();
|
|
|
+ }
|
|
|
+
|
|
|
+ /// @brief Interface Manager's fake configuration control.
|
|
|
+ IfaceMgrTestConfig iface_mgr_test_config_;
|
|
|
+
|
|
|
+};
|
|
|
+
|
|
|
+// Test that directly connected client's DHCPINFORM message is processed and
|
|
|
+// DHCPACK message is sent back.
|
|
|
+TEST_F(InformTest, directClientBroadcast) {
|
|
|
+ Dhcp4Client client;
|
|
|
+ // Configure DHCP server.
|
|
|
+ configure(INFORM_CONFIGS[0], *client.getServer());
|
|
|
+ // Request some configuration when DHCPINFORM is sent.
|
|
|
+ client.requestOptions(DHO_LOG_SERVERS, DHO_COOKIE_SERVERS);
|
|
|
+ // Preconfigure the client with the IP address.
|
|
|
+ client.createLease(IOAddress("10.0.0.56"), 600);
|
|
|
+
|
|
|
+ // Send DHCPINFORM message to the server.
|
|
|
+ ASSERT_NO_THROW(client.doInform());
|
|
|
+
|
|
|
+ // Make sure that the server responded.
|
|
|
+ ASSERT_TRUE(client.getContext().response_);
|
|
|
+ Pkt4Ptr resp = client.getContext().response_;
|
|
|
+ // Make sure that the server has responded with DHCPACK.
|
|
|
+ ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
|
|
|
+ // Response should have been unicast to the ciaddr.
|
|
|
+ EXPECT_EQ(IOAddress("10.0.0.56"), resp->getLocalAddr());
|
|
|
+ // Response must not be relayed.
|
|
|
+ EXPECT_FALSE(resp->isRelayed());
|
|
|
+ // Make sure that the server id is present.
|
|
|
+ EXPECT_EQ("10.0.0.1", client.config_.serverid_.toText());
|
|
|
+
|
|
|
+ // Make sure that the Routers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.routers_.size());
|
|
|
+ EXPECT_EQ("10.0.0.200", client.config_.routers_[0].toText());
|
|
|
+ EXPECT_EQ("10.0.0.201", client.config_.routers_[1].toText());
|
|
|
+ // Make sure that the DNS Servers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.dns_servers_.size());
|
|
|
+ EXPECT_EQ("10.0.0.202", client.config_.dns_servers_[0].toText());
|
|
|
+ EXPECT_EQ("10.0.0.203", client.config_.dns_servers_[1].toText());
|
|
|
+ // Make sure that the Log Servers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.quotes_servers_.size());
|
|
|
+ EXPECT_EQ("10.0.0.200", client.config_.routers_[0].toText());
|
|
|
+ EXPECT_EQ("10.0.0.201", client.config_.routers_[1].toText());
|
|
|
+ // Make sure that the Quotes Servers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.log_servers_.size());
|
|
|
+ EXPECT_EQ("10.0.0.202", client.config_.dns_servers_[0].toText());
|
|
|
+ EXPECT_EQ("10.0.0.203", client.config_.dns_servers_[1].toText());
|
|
|
+
|
|
|
+ // Check that we can send another DHCPINFORM message using
|
|
|
+ // different ciaddr and we will get the configuration.
|
|
|
+ client.createLease(IOAddress("10.0.0.12"), 600);
|
|
|
+ // This time do not request Quotes Servers option and it should not
|
|
|
+ // be returned.
|
|
|
+ client.requestOptions(DHO_LOG_SERVERS);
|
|
|
+
|
|
|
+ // Send DHCPINFORM.
|
|
|
+ ASSERT_NO_THROW(client.doInform());
|
|
|
+
|
|
|
+ // Make sure that the server responded.
|
|
|
+ resp = client.getContext().response_;
|
|
|
+ ASSERT_TRUE(resp);
|
|
|
+ // Make sure that the server has responded with DHCPACK.
|
|
|
+ ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
|
|
|
+ // Response should have been unicast to the ciaddr.
|
|
|
+ EXPECT_EQ(IOAddress("10.0.0.12"), resp->getLocalAddr());
|
|
|
+ // Response must not be relayed.
|
|
|
+ EXPECT_FALSE(resp->isRelayed());
|
|
|
+
|
|
|
+ // Make sure that the server id is present.
|
|
|
+ EXPECT_EQ("10.0.0.1", client.config_.serverid_.toText());
|
|
|
+ // Make sure that the Routers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.routers_.size());
|
|
|
+ EXPECT_EQ("10.0.0.200", client.config_.routers_[0].toText());
|
|
|
+ EXPECT_EQ("10.0.0.201", client.config_.routers_[1].toText());
|
|
|
+ // Make sure that the DNS Servers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.dns_servers_.size());
|
|
|
+ EXPECT_EQ("10.0.0.202", client.config_.dns_servers_[0].toText());
|
|
|
+ EXPECT_EQ("10.0.0.203", client.config_.dns_servers_[1].toText());
|
|
|
+ // Make sure that the Quotes Servers option hasn't been received.
|
|
|
+ ASSERT_TRUE(client.config_.quotes_servers_.empty());
|
|
|
+}
|
|
|
+
|
|
|
+// This test checks that the server drops DHCPINFORM message when the
|
|
|
+// source address and ciaddr is 0.
|
|
|
+TEST_F(InformTest, directClientBroadcastNoAddress) {
|
|
|
+ Dhcp4Client client;
|
|
|
+ // Configure DHCP server.
|
|
|
+ configure(INFORM_CONFIGS[0], *client.getServer());
|
|
|
+ // Request some configuration when DHCPINFORM is sent.
|
|
|
+ client.requestOptions(DHO_LOG_SERVERS, DHO_COOKIE_SERVERS);
|
|
|
+ // Send DHCPINFORM message to the server.
|
|
|
+ ASSERT_NO_THROW(client.doInform());
|
|
|
+ // Make sure that the server dropped the message.
|
|
|
+ ASSERT_FALSE(client.getContext().response_);
|
|
|
+}
|
|
|
+
|
|
|
+// Test that client's DHCPINFORM message sent to a unicast address
|
|
|
+// is received and processed by the server and that the DHCPACK is
|
|
|
+// is sent.
|
|
|
+TEST_F(InformTest, directClientUnicast) {
|
|
|
+ Dhcp4Client client;
|
|
|
+ // Configure DHCP server.
|
|
|
+ configure(INFORM_CONFIGS[0], *client.getServer());
|
|
|
+ // Preconfigure the client with the IP address.
|
|
|
+ client.createLease(IOAddress("10.0.0.56"), 600);
|
|
|
+ // Set remote address to unicast.
|
|
|
+ client.setDestAddress(IOAddress("10.0.0.1"));
|
|
|
+ // Send DHCPINFORM message to the server.
|
|
|
+ ASSERT_NO_THROW(client.doInform());
|
|
|
+ // Make sure that the server responded.
|
|
|
+ ASSERT_TRUE(client.getContext().response_);
|
|
|
+ Pkt4Ptr resp = client.getContext().response_;
|
|
|
+ // Make sure that the server has responded with DHCPACK.
|
|
|
+ ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
|
|
|
+ // Response should have been unicast to the ciaddr.
|
|
|
+ EXPECT_EQ(IOAddress("10.0.0.56"), resp->getLocalAddr());
|
|
|
+ // Response must not be relayed.
|
|
|
+ EXPECT_FALSE(resp->isRelayed());
|
|
|
+ // Make sure that the server id is present.
|
|
|
+ EXPECT_EQ("10.0.0.1", client.config_.serverid_.toText());
|
|
|
+ // Make sure that the Routers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.routers_.size());
|
|
|
+ EXPECT_EQ("10.0.0.200", client.config_.routers_[0].toText());
|
|
|
+ EXPECT_EQ("10.0.0.201", client.config_.routers_[1].toText());
|
|
|
+ // Make sure that the DNS Servers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.dns_servers_.size());
|
|
|
+ EXPECT_EQ("10.0.0.202", client.config_.dns_servers_[0].toText());
|
|
|
+ EXPECT_EQ("10.0.0.203", client.config_.dns_servers_[1].toText());
|
|
|
+}
|
|
|
+
|
|
|
+// This test checks that the server responds to the source address of the
|
|
|
+// packet received from the directly connected client if the client didn't
|
|
|
+// set the ciaddr.
|
|
|
+TEST_F(InformTest, directClientNoCiaddr) {
|
|
|
+ Dhcp4Client client;
|
|
|
+ // Configure DHCP server.
|
|
|
+ configure(INFORM_CONFIGS[0], *client.getServer());
|
|
|
+ // Preconfigure the client with the IP address.
|
|
|
+ client.createLease(IOAddress("10.0.0.56"), 600);
|
|
|
+ // Send DHCPINFORM message (with ciaddr not set) to the server.
|
|
|
+ ASSERT_NO_THROW(client.doInform(false));
|
|
|
+ // Make sure that the server responded.
|
|
|
+ ASSERT_TRUE(client.getContext().response_);
|
|
|
+ Pkt4Ptr resp = client.getContext().response_;
|
|
|
+ // Make sure that the server has responded with DHCPACK.
|
|
|
+ ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
|
|
|
+ // Response should have been unicast to the ciaddr.
|
|
|
+ EXPECT_EQ(IOAddress("10.0.0.56"), resp->getLocalAddr());
|
|
|
+ // Response must not be relayed.
|
|
|
+ EXPECT_FALSE(resp->isRelayed());
|
|
|
+ // Make sure that the server id is present.
|
|
|
+ EXPECT_EQ("10.0.0.1", client.config_.serverid_.toText());
|
|
|
+ // Make sure that the Routers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.routers_.size());
|
|
|
+ EXPECT_EQ("10.0.0.200", client.config_.routers_[0].toText());
|
|
|
+ EXPECT_EQ("10.0.0.201", client.config_.routers_[1].toText());
|
|
|
+ // Make sure that the DNS Servers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.dns_servers_.size());
|
|
|
+ EXPECT_EQ("10.0.0.202", client.config_.dns_servers_[0].toText());
|
|
|
+ EXPECT_EQ("10.0.0.203", client.config_.dns_servers_[1].toText());
|
|
|
+}
|
|
|
+
|
|
|
+// This test checks that the server receiving DHCPINFORM via relay, unicasts the
|
|
|
+// DHCPACK to the client (ciaddr).
|
|
|
+TEST_F(InformTest, relayedClient) {
|
|
|
+ Dhcp4Client client;
|
|
|
+ // Configure DHCP server.
|
|
|
+ configure(INFORM_CONFIGS[1], *client.getServer());
|
|
|
+ // Message is relayed.
|
|
|
+ client.useRelay();
|
|
|
+ // Request some configuration when DHCPINFORM is sent.
|
|
|
+ client.requestOptions(DHO_LOG_SERVERS, DHO_COOKIE_SERVERS);
|
|
|
+ // Preconfigure the client with the IP address.
|
|
|
+ client.createLease(IOAddress("192.0.2.56"), 600);
|
|
|
+ // Send DHCPINFORM message to the server.
|
|
|
+ ASSERT_NO_THROW(client.doInform());
|
|
|
+ // Make sure that the server responded.
|
|
|
+ ASSERT_TRUE(client.getContext().response_);
|
|
|
+ Pkt4Ptr resp = client.getContext().response_;
|
|
|
+ // Make sure that the server has responded with DHCPACK.
|
|
|
+ ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
|
|
|
+ // Response should have been unicast to the ciaddr.
|
|
|
+ EXPECT_EQ(IOAddress("192.0.2.56"), resp->getLocalAddr());
|
|
|
+ // Response is unicast to the client, so it must not be relayed.
|
|
|
+ EXPECT_FALSE(resp->isRelayed());
|
|
|
+ // Make sure that the server id is present.
|
|
|
+ EXPECT_EQ("10.0.0.1", client.config_.serverid_.toText());
|
|
|
+ // Make sure that the Routers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.routers_.size());
|
|
|
+ EXPECT_EQ("192.0.2.200", client.config_.routers_[0].toText());
|
|
|
+ EXPECT_EQ("192.0.2.201", client.config_.routers_[1].toText());
|
|
|
+ // Make sure that the DNS Servers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.dns_servers_.size());
|
|
|
+ EXPECT_EQ("192.0.2.202", client.config_.dns_servers_[0].toText());
|
|
|
+ EXPECT_EQ("192.0.2.203", client.config_.dns_servers_[1].toText());
|
|
|
+ // Make sure that the Log Servers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.quotes_servers_.size());
|
|
|
+ EXPECT_EQ("192.0.2.200", client.config_.routers_[0].toText());
|
|
|
+ EXPECT_EQ("192.0.2.201", client.config_.routers_[1].toText());
|
|
|
+ // Make sure that the Quotes Servers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.log_servers_.size());
|
|
|
+ EXPECT_EQ("192.0.2.202", client.config_.dns_servers_[0].toText());
|
|
|
+ EXPECT_EQ("192.0.2.203", client.config_.dns_servers_[1].toText());
|
|
|
+}
|
|
|
+
|
|
|
+// This test checks that the server can respond to the DHCPINFORM message
|
|
|
+// received via relay when the ciaddr is not set.
|
|
|
+TEST_F(InformTest, relayedClientNoCiaddr) {
|
|
|
+ Dhcp4Client client;
|
|
|
+ // Configure DHCP server.
|
|
|
+ configure(INFORM_CONFIGS[1], *client.getServer());
|
|
|
+ // Message is relayed.
|
|
|
+ client.useRelay();
|
|
|
+ // Send DHCPINFORM message to the server.
|
|
|
+ ASSERT_NO_THROW(client.doInform());
|
|
|
+ // Make sure that the server responded.
|
|
|
+ ASSERT_TRUE(client.getContext().response_);
|
|
|
+ Pkt4Ptr resp = client.getContext().response_;
|
|
|
+ // Make sure that the server has responded with DHCPACK.
|
|
|
+ ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
|
|
|
+ // Response should go through a relay as there is no ciaddr.
|
|
|
+ EXPECT_EQ(IOAddress("192.0.2.2"), resp->getLocalAddr());
|
|
|
+ EXPECT_EQ(IOAddress("192.0.2.2"), resp->getGiaddr());
|
|
|
+ EXPECT_EQ(1, resp->getHops());
|
|
|
+ // In the case when the client didn't set the ciaddr and the message
|
|
|
+ // was received via relay the server sets the Broadcast flag to help
|
|
|
+ // the relay forwarding the message (without yiaddr) to the client.
|
|
|
+ EXPECT_EQ(BOOTP_BROADCAST, resp->getFlags() & BOOTP_BROADCAST);
|
|
|
+ // Make sure that the server id is present.
|
|
|
+ EXPECT_EQ("10.0.0.1", client.config_.serverid_.toText());
|
|
|
+ // Make sure that the Routers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.routers_.size());
|
|
|
+ EXPECT_EQ("192.0.2.200", client.config_.routers_[0].toText());
|
|
|
+ EXPECT_EQ("192.0.2.201", client.config_.routers_[1].toText());
|
|
|
+ // Make sure that the DNS Servers option has been received.
|
|
|
+ ASSERT_EQ(2, client.config_.dns_servers_.size());
|
|
|
+ EXPECT_EQ("192.0.2.202", client.config_.dns_servers_[0].toText());
|
|
|
+ EXPECT_EQ("192.0.2.203", client.config_.dns_servers_[1].toText());
|
|
|
+}
|
|
|
+
|
|
|
+
|
|
|
+} // end of anonymous namespace
|