1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669 |
- // Copyright (C) 2014-2017 Internet Systems Consortium, Inc. ("ISC")
- //
- // This Source Code Form is subject to the terms of the Mozilla Public
- // License, v. 2.0. If a copy of the MPL was not distributed with this
- // file, You can obtain one at http://mozilla.org/MPL/2.0/.
- #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 <dhcpsrv/cfgmgr.h>
- #include <dhcpsrv/host.h>
- #include <dhcpsrv/host_mgr.h>
- #include <dhcpsrv/subnet_id.h>
- #include <dhcpsrv/testutils/cql_schema.h>
- #include <dhcpsrv/testutils/mysql_schema.h>
- #include <dhcpsrv/testutils/pgsql_schema.h>
- #include <dhcp4/tests/dhcp4_test_utils.h>
- #include <dhcp4/tests/dhcp4_client.h>
- #include <boost/shared_ptr.hpp>
- #include <stats/stats_mgr.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 DORA 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.
- ///
- /// - Configuration 2:
- /// - Use for testing simple scenarios with host reservations
- /// - 1 subnet: 10.0.0.0/24
- /// - One reservation for the client using MAC address:
- /// aa:bb:cc:dd:ee:ff, reserved address 10.0.0.7
- ///
- /// - Configuration 3:
- /// - Use for testing match-client-id flag
- /// - 1 subnet: 10.0.0.0/24
- /// - 1 pool: 10.0.0.10-10.0.0.100
- /// - match-client-id flag is set to false, thus the server
- /// uses HW address for lease lookup, rather than client id
- ///
- /// - Configuration 4:
- /// - Used for testing host reservations where circuit-id takes precedence
- /// over hw-address, and the hw-address takes precedence over duid.
- /// - 1 subnet: 10.0.0.0/24
- /// - 3 reservations for this subnet:
- /// - IP address 10.0.0.7 for HW address aa:bb:cc:dd:ee:ff
- /// - IP address 10.0.0.8 for DUID 01:02:03:04:05
- /// - IP address 10.0.0.9 for circuit-id 'charter950'
- /// - IP address 10.0.0.1 for client-id
- ///
- /// - Configuration 5:
- /// - The same as configuration 4, but using the following order of
- /// host-reservation-identifiers: duid, circuit-id, hw-address,
- /// client-id.
- ///
- /// - Configuration 6:
- /// - This configuration provides reservations for next-server,
- /// server-hostname and boot-file-name value.
- /// - 1 subnet: 10.0.0.0/24
- /// - 1 reservation for this subnet:
- /// - Client's HW address: aa:bb:cc:dd:ee:ff
- /// - next-server = 10.0.0.7
- /// - server name = "some-name.example.org"
- /// - boot-file-name = "bootfile.efi"
- ///
- /// - Configuration 7:
- /// - Simple configuration with a single subnet and single pool
- /// - Using MySQL lease database backend to store leases
- ///
- /// - Configuration 8:
- /// - Simple configuration with a single subnet and single pool
- /// - Using PostgreSQL lease database backend to store leases
- ///
- /// - Configuration 9:
- /// - Simple configuration with a single subnet and single pool
- /// - Using Cassandra lease database backend to store leases
- const char* DORA_CONFIGS[] = {
- // Configuration 0
- "{ \"interfaces-config\": {"
- " \"interfaces\": [ \"*\" ]"
- "},"
- "\"valid-lifetime\": 600,"
- "\"subnet4\": [ { "
- " \"subnet\": \"10.0.0.0/24\", "
- " \"id\": 1,"
- " \"pools\": [ { \"pool\": \"10.0.0.10-10.0.0.100\" } ],"
- " \"option-data\": [ {"
- " \"name\": \"routers\","
- " \"data\": \"10.0.0.200,10.0.0.201\""
- " },"
- " {"
- " \"name\": \"domain-name-servers\","
- " \"data\": \"10.0.0.202,10.0.0.203\""
- " },"
- " {"
- " \"name\": \"log-servers\","
- " \"data\": \"10.0.0.200,10.0.0.201\""
- " },"
- " {"
- " \"name\": \"cookie-servers\","
- " \"data\": \"10.0.0.202,10.0.0.203\""
- " } ]"
- " } ]"
- "}",
- // Configuration 1
- "{ \"interfaces-config\": {"
- " \"interfaces\": [ \"*\" ]"
- "},"
- "\"valid-lifetime\": 600,"
- "\"subnet4\": [ { "
- " \"subnet\": \"192.0.2.0/24\", "
- " \"option-data\": [ {"
- " \"name\": \"routers\","
- " \"data\": \"192.0.2.200,192.0.2.201\""
- " },"
- " {"
- " \"name\": \"domain-name-servers\","
- " \"data\": \"192.0.2.202,192.0.2.203\""
- " },"
- " {"
- " \"name\": \"log-servers\","
- " \"data\": \"10.0.0.200,10.0.0.201\""
- " },"
- " {"
- " \"name\": \"cookie-servers\","
- " \"data\": \"10.0.0.202,10.0.0.203\""
- " } ]"
- " } ]"
- "}",
- // Configuration 2
- "{ \"interfaces-config\": {"
- " \"interfaces\": [ \"*\" ]"
- "},"
- "\"valid-lifetime\": 600,"
- "\"subnet4\": [ { "
- " \"subnet\": \"10.0.0.0/24\", "
- " \"pools\": [ { \"pool\": \"10.0.0.10-10.0.0.100\" } ],"
- " \"reservations\": [ "
- " {"
- " \"hw-address\": \"aa:bb:cc:dd:ee:ff\","
- " \"ip-address\": \"10.0.0.7\""
- " },"
- " {"
- " \"duid\": \"01:02:03:04:05\","
- " \"ip-address\": \"10.0.0.8\""
- " },"
- " {"
- " \"circuit-id\": \"'charter950'\","
- " \"ip-address\": \"10.0.0.9\""
- " },"
- " {"
- " \"client-id\": \"01:11:22:33:44:55:66\","
- " \"ip-address\": \"10.0.0.1\""
- " }"
- " ]"
- "} ]"
- "}",
- // Configuration 3
- "{ \"interfaces-config\": {"
- " \"interfaces\": [ \"*\" ]"
- "},"
- "\"valid-lifetime\": 600,"
- "\"match-client-id\": false,"
- "\"subnet4\": [ { "
- " \"subnet\": \"10.0.0.0/24\", "
- " \"id\": 1,"
- " \"pools\": [ { \"pool\": \"10.0.0.10-10.0.0.100\" } ],"
- " \"option-data\": [ {"
- " \"name\": \"routers\","
- " \"data\": \"10.0.0.200,10.0.0.201\""
- " } ]"
- " } ]"
- "}",
- // Configuration 4
- "{ \"interfaces-config\": {"
- " \"interfaces\": [ \"*\" ]"
- "},"
- "\"host-reservation-identifiers\": [ \"circuit-id\", \"hw-address\","
- " \"duid\", \"client-id\" ],"
- "\"valid-lifetime\": 600,"
- "\"subnet4\": [ { "
- " \"subnet\": \"10.0.0.0/24\", "
- " \"pools\": [ { \"pool\": \"10.0.0.10-10.0.0.100\" } ],"
- " \"reservations\": [ "
- " {"
- " \"hw-address\": \"aa:bb:cc:dd:ee:ff\","
- " \"ip-address\": \"10.0.0.7\""
- " },"
- " {"
- " \"duid\": \"01:02:03:04:05\","
- " \"ip-address\": \"10.0.0.8\""
- " },"
- " {"
- " \"circuit-id\": \"'charter950'\","
- " \"ip-address\": \"10.0.0.9\""
- " },"
- " {"
- " \"client-id\": \"01:11:22:33:44:55:66\","
- " \"ip-address\": \"10.0.0.1\""
- " }"
- " ]"
- "} ]"
- "}",
- // Configuration 5
- "{ \"interfaces-config\": {"
- " \"interfaces\": [ \"*\" ]"
- "},"
- "\"host-reservation-identifiers\": [ \"duid\", \"client-id\","
- " \"circuit-id\", \"hw-address\" ],"
- "\"valid-lifetime\": 600,"
- "\"subnet4\": [ { "
- " \"subnet\": \"10.0.0.0/24\", "
- " \"pools\": [ { \"pool\": \"10.0.0.10-10.0.0.100\" } ],"
- " \"reservations\": [ "
- " {"
- " \"hw-address\": \"aa:bb:cc:dd:ee:ff\","
- " \"ip-address\": \"10.0.0.7\""
- " },"
- " {"
- " \"duid\": \"01:02:03:04:05\","
- " \"ip-address\": \"10.0.0.8\""
- " },"
- " {"
- " \"circuit-id\": \"'charter950'\","
- " \"ip-address\": \"10.0.0.9\""
- " },"
- " {"
- " \"client-id\": \"01:11:22:33:44:55:66\","
- " \"ip-address\": \"10.0.0.1\""
- " }"
- " ]"
- "} ]"
- "}",
- // Configuration 6
- "{ \"interfaces-config\": {"
- " \"interfaces\": [ \"*\" ]"
- "},"
- "\"valid-lifetime\": 600,"
- "\"next-server\": \"10.0.0.1\","
- "\"server-hostname\": \"nohost\","
- "\"boot-file-name\": \"nofile\","
- "\"subnet4\": [ { "
- " \"subnet\": \"10.0.0.0/24\", "
- " \"pools\": [ { \"pool\": \"10.0.0.10-10.0.0.100\" } ],"
- " \"reservations\": [ "
- " {"
- " \"hw-address\": \"aa:bb:cc:dd:ee:ff\","
- " \"next-server\": \"10.0.0.7\","
- " \"server-hostname\": \"some-name.example.org\","
- " \"boot-file-name\": \"bootfile.efi\""
- " }"
- " ]"
- "} ]"
- "}",
- // Configuration 7
- "{ \"interfaces-config\": {"
- " \"interfaces\": [ \"*\" ]"
- "},"
- "\"lease-database\": {"
- " \"type\": \"mysql\","
- " \"name\": \"keatest\","
- " \"user\": \"keatest\","
- " \"password\": \"keatest\""
- "},"
- "\"valid-lifetime\": 600,"
- "\"subnet4\": [ { "
- " \"subnet\": \"10.0.0.0/24\", "
- " \"id\": 1,"
- " \"pools\": [ { \"pool\": \"10.0.0.10-10.0.0.100\" } ]"
- " } ]"
- "}",
- // Configuration 8
- "{ \"interfaces-config\": {"
- " \"interfaces\": [ \"*\" ]"
- "},"
- "\"lease-database\": {"
- " \"type\": \"postgresql\","
- " \"name\": \"keatest\","
- " \"user\": \"keatest\","
- " \"password\": \"keatest\""
- "},"
- "\"valid-lifetime\": 600,"
- "\"subnet4\": [ { "
- " \"subnet\": \"10.0.0.0/24\", "
- " \"id\": 1,"
- " \"pools\": [ { \"pool\": \"10.0.0.10-10.0.0.100\" } ]"
- " } ]"
- "}",
- // Configuration 9
- "{ \"interfaces-config\": {"
- " \"interfaces\": [ \"*\" ]"
- "},"
- "\"lease-database\": {"
- " \"type\": \"cql\","
- " \"name\": \"keatest\","
- " \"user\": \"keatest\","
- " \"password\": \"keatest\""
- "},"
- "\"valid-lifetime\": 600,"
- "\"subnet4\": [ { "
- " \"subnet\": \"10.0.0.0/24\", "
- " \"id\": 1,"
- " \"pools\": [ { \"pool\": \"10.0.0.10-10.0.0.100\" } ]"
- " } ]"
- "}"
- };
- /// @brief Test fixture class for testing 4-way (DORA) exchanges.
- ///
- /// @todo Currently there is a limit number of test cases covered here.
- /// In the future it is planned that the tests from the
- /// dhcp4_srv_unittest.cc will be migrated here and will use the
- /// @c Dhcp4Client class.
- class DORATest : public Dhcpv4SrvTest {
- public:
- /// @brief Constructor.
- ///
- /// Sets up fake interfaces.
- DORATest()
- : Dhcpv4SrvTest(),
- iface_mgr_test_config_(true) {
- IfaceMgr::instance().openSockets4();
- // Let's wipe all existing statistics.
- isc::stats::StatsMgr::instance().removeAll();
- }
- /// @brief Destructor.
- ///
- /// Cleans up statistics after the test.
- ~DORATest() {
- // Let's wipe all existing statistics.
- isc::stats::StatsMgr::instance().removeAll();
- }
- /// @brief Test that server returns the same lease for the client which is
- /// sometimes using client identifier, sometimes not.
- ///
- /// This test checks the server's behavior in the following scenario:
- /// - Client identifies itself to the server using HW address, and may use
- /// client identifier.
- /// - Client performs the 4-way exchange and obtains a lease from the server.
- /// - If the client identifier was in use when the client has acquired the lease,
- /// the client uses null client identifier in the next exchange with the server.
- /// - If the client identifier was not in use when the client has acquired the
- /// lease, the client uses client identifier in the next exchange with the
- /// server.
- /// - When the client contacts the server for the second time using the
- /// DHCPDISCOVER the server determines (using HW address) that the client
- /// already has a lease and returns this lease to the client.
- /// - The client renews the existing lease.
- ///
- /// @param clientid_a Client identifier when the client initially allocates
- /// the lease. An empty value means "no client identifier".
- /// @param clientid_b Client identifier when the client sends the DHCPDISCOVER
- /// and then DHCPREQUEST to renew lease.
- void oneAllocationOverlapTest(const std::string& clientid_a,
- const std::string& clientid_b);
- /// @brief Test that the client using the same hardware address but
- /// multiple client identifiers will obtain multiple leases.
- ///
- /// This reflects the scenario of the OS installation over the network
- /// when BIOS, installer and the host request DHCPv4 lease assignment
- /// using the same MAC address/interface but generating different
- /// client identifiers.
- ///
- /// @param config_index Index of the configuration within the
- /// @c DORA_CONFIGS array.
- void testMultiStageBoot(const unsigned int config_index);
- /// @brief Interface Manager's fake configuration control.
- IfaceMgrTestConfig iface_mgr_test_config_;
- };
- /// This test verifies that the client in the SELECTING state can get
- /// an address when it doesn't request any specific address in the
- /// DHCPDISCOVER message.
- TEST_F(DORATest, selectingDoNotRequestAddress) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Configure DHCP server.
- configure(DORA_CONFIGS[0], *client.getServer());
- // Perform 4-way exchange with the server but to not request any
- // specific address in the DHCPDISCOVER message.
- ASSERT_NO_THROW(client.doDORA());
- // 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 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 client has got the lease with the requested address.
- ASSERT_NE(client.config_.lease_.addr_.toText(), "0.0.0.0");
- }
- /// This test verifies that multiple clients may use the DHCPv4 server
- /// and obtain unique leases.
- TEST_F(DORATest, selectingMultipleClients) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Configure DHCP server.
- configure(DORA_CONFIGS[0], *client.getServer());
- // Get the first lease.
- ASSERT_NO_THROW(client.doDORA());
- // Make sure that the server responded.
- Pkt4Ptr resp = client.getContext().response_;
- ASSERT_TRUE(resp);
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Store the lease.
- Lease4 lease1 = client.config_.lease_;
- // Get the lease for a different client.
- client.modifyHWAddr();
- ASSERT_NO_THROW(client.doDORA());
- // Make sure that the server responded.
- resp = client.getContext().response_;
- ASSERT_TRUE(resp);
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Store the lease.
- Lease4 lease2 = client.config_.lease_;
- // Get the lease for a different client.
- client.modifyHWAddr();
- ASSERT_NO_THROW(client.doDORA());
- // Make sure that the server responded.
- resp = client.getContext().response_;
- ASSERT_TRUE(resp);
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Store the lease.
- Lease4 lease3 = client.config_.lease_;
- // Make sure that unique addresses have been assigned.
- EXPECT_NE(lease1.addr_, lease2.addr_);
- EXPECT_NE(lease2.addr_, lease3.addr_);
- EXPECT_NE(lease1.addr_, lease3.addr_);
- }
- // This test verifies that the client in a SELECTING state can request
- // a specific address and that this address will be assigned when
- // available. It also tests that if the client requests an address which
- // is in use the client will get a different address.
- TEST_F(DORATest, selectingRequestAddress) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Configure DHCP server.
- configure(DORA_CONFIGS[0], *client.getServer());
- // Perform 4-way exchange with the server.
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("10.0.0.50"))));
- // 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 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 client has got the lease with the requested address.
- ASSERT_EQ("10.0.0.50", client.config_.lease_.addr_.toText());
- // Simulate different client requesting the same address.
- client.modifyHWAddr();
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("10.0.0.50"))));
- resp = client.getContext().response_;
- // Make sure that the server responded.
- ASSERT_TRUE(resp);
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // 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 client has got some address.
- EXPECT_NE(client.config_.lease_.addr_.toText(), "0.0.0.0");
- // Make sure that the client has got a different address than requested
- // as the requested one is already in use.
- EXPECT_NE(client.config_.lease_.addr_.toText(), "10.0.0.50");
- }
- // This test verifies that the client will get the address that it has
- // been allocated when the client requests a different address.
- TEST_F(DORATest, selectingRequestNonMatchingAddress) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Configure DHCP server.
- configure(DORA_CONFIGS[0], *client.getServer());
- // Perform 4-way exchange with the server.
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("10.0.0.50"))));
- // 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 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 client has got the lease with the requested address.
- ASSERT_EQ("10.0.0.50", client.config_.lease_.addr_.toText());
- // Let's request a different address. The server should respond with
- // the one that the client already has allocated.
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("10.0.0.80"))));
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // 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 client has got the lease with the address that
- // the client has recorded in the lease database.
- EXPECT_EQ("10.0.0.50", client.config_.lease_.addr_.toText());
- }
- // Test that the client in the INIT-REBOOT state can request the IP
- // address it has and the address is returned. Also, check that if
- // if the client requests invalid address the server sends a DHCPNAK.
- TEST_F(DORATest, initRebootRequest) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Configure DHCP server.
- configure(DORA_CONFIGS[0], *client.getServer());
- client.includeClientId("11:22");
- // Obtain a lease from the server using the 4-way exchange.
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("10.0.0.50"))));
- // 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 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 client has got the lease with the requested address.
- ASSERT_EQ("10.0.0.50", client.config_.lease_.addr_.toText());
- // Client has a lease in the database. Let's transition the client
- // to the INIT_REBOOT state so as the client can request the cached
- // lease using the DHCPREQUEST message.
- client.setState(Dhcp4Client::INIT_REBOOT);
- ASSERT_NO_THROW(client.doRequest());
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // 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 client has got the lease with the requested address.
- ASSERT_EQ("10.0.0.50", client.config_.lease_.addr_.toText());
- // Try to request a different address than the client has. The server
- // should respond with DHCPNAK.
- client.config_.lease_.addr_ = IOAddress("10.0.0.30");
- ASSERT_NO_THROW(client.doRequest());
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- EXPECT_EQ(DHCPNAK, static_cast<int>(resp->getType()));
- // Change client identifier. The server should treat the request
- // as a request from unknown client and ignore it.
- client.includeClientId("12:34");
- ASSERT_NO_THROW(client.doRequest());
- ASSERT_FALSE(client.getContext().response_);
- // Now let's fix the IP address. The client identifier is still
- // invalid so the message should be dropped.
- client.config_.lease_.addr_ = IOAddress("10.0.0.50");
- ASSERT_NO_THROW(client.doRequest());
- ASSERT_FALSE(client.getContext().response_);
- // Restore original client identifier.
- client.includeClientId("11:22");
- // Try to request from a different HW address. This should be successful
- // because the client identifier matches.
- client.modifyHWAddr();
- ASSERT_NO_THROW(client.doRequest());
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Make sure that the client has got the lease with the requested address.
- ASSERT_EQ("10.0.0.50", client.config_.lease_.addr_.toText());
- }
- // Check that the ciaddr returned by the server is correct for DHCPOFFER and
- // DHCPNAK according to RFC2131, section 4.3.1.
- TEST_F(DORATest, ciaddr) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Configure DHCP server.
- configure(DORA_CONFIGS[0], *client.getServer());
- // Force ciaddr of Discover message to be non-zero.
- client.ciaddr_.specify(IOAddress("10.0.0.50"));
- // Obtain a lease from the server using the 4-way exchange.
- ASSERT_NO_THROW(client.doDiscover(boost::shared_ptr<
- IOAddress>(new IOAddress("10.0.0.50"))));
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- Pkt4Ptr resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPOFFER.
- ASSERT_EQ(DHCPOFFER, static_cast<int>(resp->getType()));
- // Make sure ciaddr is not set for DHCPOFFER.
- EXPECT_EQ("0.0.0.0", resp->getCiaddr().toText());
- // Obtain a lease from the server using the 4-way exchange.
- ASSERT_NO_THROW(client.doRequest());
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Let's transition the client to Renewing state.
- client.setState(Dhcp4Client::RENEWING);
- // Set the unicast destination address to indicate that it is a renewal.
- client.setDestAddress(IOAddress("10.0.0.1"));
- ASSERT_NO_THROW(client.doRequest());
- // The client is sending invalid ciaddr so the server should send a NAK.
- resp = client.getContext().response_;
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // For DHCPACK the ciaddr may be 0 or may be set to the ciaddr value
- // from the client's message. Kea sets it to the latter.
- EXPECT_EQ("10.0.0.50", resp->getCiaddr().toText());
- // Replace the address held by the client. The client will request
- // the assignment of this address but the server has a different
- // address for this client.
- client.ciaddr_.specify(IOAddress("192.168.0.30"));
- ASSERT_NO_THROW(client.doRequest());
- // The client is sending invalid ciaddr so the server should send a NAK.
- resp = client.getContext().response_;
- ASSERT_EQ(DHCPNAK, static_cast<int>(resp->getType()));
- // For DHCPNAK the ciaddr is always 0 (should not be copied) from the
- // client's message.
- EXPECT_EQ("0.0.0.0", resp->getCiaddr().toText());
- }
- void
- DORATest::oneAllocationOverlapTest(const std::string& clientid_a,
- const std::string& clientid_b) {
- // Allocate a lease by client using the 4-way exchange.
- Dhcp4Client client(Dhcp4Client::SELECTING);
- client.includeClientId(clientid_a);
- client.setHWAddress("01:02:03:04:05:06");
- configure(DORA_CONFIGS[0], *client.getServer());
- ASSERT_NO_THROW(client.doDORA());
- // 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()));
- Lease4Ptr lease_a = LeaseMgrFactory::instance().getLease4(client.config_.lease_.addr_);
- ASSERT_TRUE(lease_a);
- // Remember the allocated address.
- IOAddress yiaddr = lease_a->addr_;
- // Change client identifier. If parameters clientid_a and clientid_b
- // are specified correctly, this removes the client identifier from
- // client's requests if the lease has been acquired with the client
- // identifier, or adds the client identifier otherwise.
- client.includeClientId(clientid_b);
- // Check if the server will offer the same address.
- ASSERT_NO_THROW(client.doDiscover());
- resp = client.getContext().response_;
- ASSERT_TRUE(resp);
- EXPECT_EQ(yiaddr, resp->getYiaddr());
- // Client should also be able to renew its address.
- client.setState(Dhcp4Client::RENEWING);
- ASSERT_NO_THROW(client.doRequest());
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- ASSERT_EQ(yiaddr, client.config_.lease_.addr_);
- }
- // This test checks the server behavior in the following situation:
- // - Client A identifies itself to the server using client identifier
- // and the hardware address and requests allocation of the new lease.
- // - Server allocates the lease to the client.
- // - Client B has the same hardware address but is using a different
- // client identifier then Client A.
- // - Client B sends DHCPDISCOVER.
- // - Server should determine that the client B is not client A, because
- // it is using a different client identifier, even though they use the
- // same HW address. As a consequence, the server should offer a
- // different address to the client B.
- // - The client B performs the 4-way exchange again and the server
- // allocates a new address to the client, which should be different
- // than the address used by the client A.
- // - Client B is in the renewing state and it successfully renews its
- // address.
- // - Client A also renews its address successfully.
- TEST_F(DORATest, twoAllocationsOverlap) {
- // Allocate a lease by client A using the 4-way exchange.
- Dhcp4Client client_a(Dhcp4Client::SELECTING);
- client_a.includeClientId("12:34");
- client_a.setHWAddress("01:02:03:04:05:06");
- configure(DORA_CONFIGS[0], *client_a.getServer());
- ASSERT_NO_THROW(client_a.doDORA());
- // Make sure that the server responded.
- ASSERT_TRUE(client_a.getContext().response_);
- Pkt4Ptr resp_a = client_a.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp_a->getType()));
- // Make sure that the lease has been recorded by the server.
- Lease4Ptr lease_a = LeaseMgrFactory::instance().getLease4(client_a.config_.lease_.addr_);
- ASSERT_TRUE(lease_a);
- // Create client B.
- Dhcp4Client client_b(client_a.getServer(), Dhcp4Client::SELECTING);
- client_b.setHWAddress("01:02:03:04:05:06");
- client_b.includeClientId("45:67");
- // Send DHCPDISCOVER and expect the response.
- ASSERT_NO_THROW(client_b.doDiscover());
- Pkt4Ptr resp_b = client_b.getContext().response_;
- // Make sure that the server has responded with DHCPOFFER.
- ASSERT_EQ(DHCPOFFER, static_cast<int>(resp_b->getType()));
- // The offered address should be different than the address which
- // was obtained by the client A.
- ASSERT_NE(resp_b->getYiaddr(), client_a.config_.lease_.addr_);
- // Make sure that the client A lease hasn't been modified.
- lease_a = LeaseMgrFactory::instance().getLease4(client_a.config_.lease_.addr_);
- ASSERT_TRUE(lease_a);
- // Now that we know that the server will avoid assigning the same
- // address that the client A has, use the 4-way exchange to actually
- // allocate some address.
- ASSERT_NO_THROW(client_b.doDORA());
- // Make sure that the server responded.
- ASSERT_TRUE(client_b.getContext().response_);
- resp_b = client_b.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp_b->getType()));
- // Again, make sure the assigned addresses are different.
- ASSERT_NE(client_b.config_.lease_.addr_, client_a.config_.lease_.addr_);
- // Make sure that the client A still has a lease.
- lease_a = LeaseMgrFactory::instance().getLease4(client_a.config_.lease_.addr_);
- ASSERT_TRUE(lease_a);
- // Make sure that the client B has a lease.
- Lease4Ptr lease_b = LeaseMgrFactory::instance().getLease4(client_b.config_.lease_.addr_);
- ASSERT_TRUE(lease_b);
- // Client B should be able to renew its address.
- client_b.setState(Dhcp4Client::RENEWING);
- ASSERT_NO_THROW(client_b.doRequest());
- ASSERT_TRUE(client_b.getContext().response_);
- resp_b = client_b.getContext().response_;
- ASSERT_EQ(DHCPACK, static_cast<int>(resp_b->getType()));
- ASSERT_NE(client_b.config_.lease_.addr_, client_a.config_.lease_.addr_);
- // Client A should also be able to renew its address.
- client_a.setState(Dhcp4Client::RENEWING);
- ASSERT_NO_THROW(client_a.doRequest());
- ASSERT_TRUE(client_a.getContext().response_);
- resp_b = client_a.getContext().response_;
- ASSERT_EQ(DHCPACK, static_cast<int>(resp_b->getType()));
- ASSERT_NE(client_a.config_.lease_.addr_, client_b.config_.lease_.addr_);
- }
- // This test checks the server behavior in the following situation:
- // - Client A identifies itself to the server using the hardware address
- // and client identifier.
- // - Client A performs the 4-way exchange and obtains a lease from the server.
- // - Client B uses the same HW address as the client A, but it doesn't use
- // the client identifier.
- // - Client B sends the DHCPDISCOVER to the server.
- // The server determines that there is a lease for the client A using the
- // same HW address as the client B. Server discards the client's message and
- // doesn't offer the lease for the client B to prevent allocation of the
- // lease without a unique identifier.
- // - The client sends the DHCPREQUEST and the server sends the DHCPNAK for the
- // same reason.
- // - Client A renews its address successfully.
- TEST_F(DORATest, oneAllocationOverlap1) {
- oneAllocationOverlapTest("12:34", "");
- }
- // This test is similar to oneAllocationOverlap2 but this time the client A
- // uses no client identifier, and the client B uses the HW address and the
- // client identifier. The server behaves as previously.
- TEST_F(DORATest, oneAllocationOverlap2) {
- oneAllocationOverlapTest("", "12:34");
- }
- // This is a simple test for the host reservation. It creates a reservation
- // for an address for a single client, identified by the HW address. The
- // test verifies that the client using this HW address will obtain a
- // lease for the reserved address. It also checks that the client using
- // a different HW address will obtain an address from the dynamic pool.
- TEST_F(DORATest, reservation) {
- // Client A is a one which will have a reservation.
- Dhcp4Client clientA(Dhcp4Client::SELECTING);
- // Set explicit HW address so as it matches the reservation in the
- // configuration used below.
- clientA.setHWAddress("aa:bb:cc:dd:ee:ff");
- // Configure DHCP server.
- configure(DORA_CONFIGS[2], *clientA.getServer());
- // Client A performs 4-way exchange and should obtain a reserved
- // address.
- ASSERT_NO_THROW(clientA.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // Make sure that the server responded.
- ASSERT_TRUE(clientA.getContext().response_);
- Pkt4Ptr resp = clientA.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Make sure that the client has got the lease for the reserved address.
- ASSERT_EQ("10.0.0.7", clientA.config_.lease_.addr_.toText());
- // Client B uses the same server as Client A.
- Dhcp4Client clientB(clientA.getServer(), Dhcp4Client::SELECTING);
- // Client B has no reservation so it should get the lease from
- // the dynamic pool.
- ASSERT_NO_THROW(clientB.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // Make sure that the server responded.
- ASSERT_TRUE(clientB.getContext().response_);
- resp = clientB.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Obtain the subnet to which the returned address belongs.
- Subnet4Ptr subnet = CfgMgr::instance().getCurrentCfg()->getCfgSubnets4()->
- selectSubnet(clientB.config_.lease_.addr_);
- ASSERT_TRUE(subnet);
- // Make sure that the address has been allocated from the dynamic pool.
- ASSERT_TRUE(subnet->inPool(Lease::TYPE_V4, clientB.config_.lease_.addr_));
- }
- // This test checks that it is possible to make a reservation by
- // DUID carried in the Client Identifier option.
- TEST_F(DORATest, reservationByDUID) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Use relay agent.
- client.useRelay(true, IOAddress("10.0.0.1"), IOAddress("10.0.0.2"));
- // Modify HW address so as the server doesn't assign reserved
- // address by HW address.
- client.modifyHWAddr();
- // Specify DUID for which address 10.0.0.8 is reserved.
- // The value specified as client id includes:
- // - FF is a client identifier type for DUID,
- // - 45454545 - represents 4 bytes for IAID
- // - 01:02:03:04:05 - is an actual DUID for which there is a
- // reservation.
- client.includeClientId("FF:45:45:45:45:01:02:03:04:05");
- // Configure DHCP server.
- configure(DORA_CONFIGS[2], *client.getServer());
- // Client A performs 4-way exchange and should obtain a reserved
- // address.
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // 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()));
- // Make sure that the client has got the lease for the reserved address.
- ASSERT_EQ("10.0.0.8", client.config_.lease_.addr_.toText());
- }
- // This test checks that it is possible to make a reservation by
- // circuit-id inserted by the relay agent.
- TEST_F(DORATest, reservationByCircuitId) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Use relay agent so as the circuit-id can be inserted.
- client.useRelay(true, IOAddress("10.0.0.1"), IOAddress("10.0.0.2"));
- // Specify circuit-id.
- client.setCircuitId("charter950");
- // Configure DHCP server.
- configure(DORA_CONFIGS[2], *client.getServer());
- // Client A performs 4-way exchange and should obtain a reserved
- // address.
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // 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()));
- // Make sure that the client has got the lease for the reserved address.
- ASSERT_EQ("10.0.0.9", client.config_.lease_.addr_.toText());
- }
- // This test checks that it is possible to make a reservation by
- // client-id.
- TEST_F(DORATest, reservationByClientId) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Use relay agent to make sure that the desired subnet is
- // selected for our client.
- client.useRelay(true, IOAddress("10.0.0.20"), IOAddress("10.0.0.21"));
- // Specify client identifier.
- client.includeClientId("01:11:22:33:44:55:66");
- // Configure DHCP server.
- configure(DORA_CONFIGS[2], *client.getServer());
- // Client A performs 4-way exchange and should obtain a reserved
- // address.
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // 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()));
- // Make sure that the client has got the lease for the reserved address.
- ASSERT_EQ("10.0.0.1", client.config_.lease_.addr_.toText());
- }
- // This test verifies that order in which host identifiers are used to
- // retrieve host reservations can be controlled.
- TEST_F(DORATest, hostIdentifiersOrder) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- client.setHWAddress("aa:bb:cc:dd:ee:ff");
- // Use relay agent so as the circuit-id can be inserted.
- client.useRelay(true, IOAddress("10.0.0.1"), IOAddress("10.0.0.2"));
- // Specify DUID for which address 10.0.0.8 is reserved.
- // The value specified as client id includes:
- // - FF is a client identifier type for DUID,
- // - 45454545 - represents 4 bytes for IAID
- // - 01:02:03:04:05 - is an actual DUID for which there is a
- // reservation.
- client.includeClientId("FF:45:45:45:45:01:02:03:04:05");
- // Specify circuit-id.
- client.setCircuitId("charter950");
- // Configure DHCP server.
- configure(DORA_CONFIGS[2], *client.getServer());
- // Perform 4-way exchange to obtain reserved address.
- // The client has in fact two reserved addresses, but the one assigned
- // should be by hw-address.
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // 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()));
- // Make sure that the client has got the lease for the reserved address.
- ASSERT_EQ("10.0.0.7", client.config_.lease_.addr_.toText());
- // Reconfigure the server to change the preference order of the
- // host identifiers. The 'circuit-id' should now take precedence over
- // the hw-address, duid and client-id.
- configure(DORA_CONFIGS[4], *client.getServer());
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Make sure that the client has got the lease for the reserved address.
- ASSERT_EQ("10.0.0.9", client.config_.lease_.addr_.toText());
- // Reconfigure the server to change the preference order of the
- // host identifiers. The 'duid' should now take precedence over
- // the client-id, hw-address and circuit-id
- configure(DORA_CONFIGS[5], *client.getServer());
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Make sure that the client has got the lease for the reserved address.
- ASSERT_EQ("10.0.0.8", client.config_.lease_.addr_.toText());
- // Replace the client identifier with the one for which address
- // 10.0.0.1 is reserved. Because the DUID is a special type of
- // client identifier, this change effectively removes the association
- // of the client with the DUID for which address 10.0.0.8 is reserved.
- // The next identifier type to be used by the server (after DUID) is
- // client-id and thus the server should assign address 10.0.0.1.
- client.includeClientId("01:11:22:33:44:55:66");
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Make sure that the client has got the lease for the reserved address.
- ASSERT_EQ("10.0.0.1", client.config_.lease_.addr_.toText());
- }
- // This test checks that setting the match-client-id value to false causes
- // the server to ignore changing client identifier when the client is
- // using consistent HW address.
- TEST_F(DORATest, ignoreChangingClientId) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Configure DHCP server.
- configure(DORA_CONFIGS[3], *client.getServer());
- client.includeClientId("12:12");
- // Obtain the lease using 4-way exchange.
- ASSERT_NO_THROW(client.doDORA());
- // 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()));
- EXPECT_FALSE(client.config_.lease_.client_id_);
- // Remember address which the client has obtained.
- IOAddress leased_address = client.config_.lease_.addr_;
- // Modify client id. Because we have set the configuration flag which
- // forces the server to lookup leases using the HW address, the
- // client id modification should not matter and the client should
- // obtain the same lease.
- client.includeClientId("14:14");
- // Obtain the lease using 4-way exchange.
- ASSERT_NO_THROW(client.doDORA());
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Make sure that the server assigned the same address, even though the
- // client id has changed.
- EXPECT_EQ(leased_address, client.config_.lease_.addr_);
- // Check that the client id is not present in the lease.
- EXPECT_FALSE(client.config_.lease_.client_id_);
- }
- // This test checks that the match-client-id parameter doesn't have
- // effect on the lease lookup using the HW address.
- TEST_F(DORATest, changingHWAddress) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Configure DHCP server.
- configure(DORA_CONFIGS[3], *client.getServer());
- client.includeClientId("12:12");
- client.setHWAddress("00:01:02:03:04:05");
- // Obtain the lease using 4-way exchange.
- ASSERT_NO_THROW(client.doDORA());
- // 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()));
- // Check that the client id is not present in the lease.
- EXPECT_FALSE(client.config_.lease_.client_id_);
- // Remember address which the client has obtained.
- IOAddress leased_address = client.config_.lease_.addr_;
- // Modify HW address but leave client id in place. The value of the
- // match-client-id set to false must not have any effect on the
- // case when the HW address is changing. In such case the server will
- // allocate the new address for the client.
- client.setHWAddress("01:01:01:01:01:01");
- // Obtain a lease.
- ASSERT_NO_THROW(client.doDORA());
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Client must assign different address because the client id is
- // ignored and the HW address was changed.
- EXPECT_NE(client.config_.lease_.addr_, leased_address);
- // Check that the client id is not present in the lease.
- EXPECT_FALSE(client.config_.lease_.client_id_);
- }
- // This test verifies that the server assigns reserved values for the
- // siaddr, sname and file fields carried within DHCPv4 message.
- TEST_F(DORATest, messageFieldsReservations) {
- // Client has a reservation.
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Set explicit HW address so as it matches the reservation in the
- // configuration used below.
- client.setHWAddress("aa:bb:cc:dd:ee:ff");
- // Configure DHCP server.
- configure(DORA_CONFIGS[6], *client.getServer());
- // Client performs 4-way exchange and should obtain a reserved
- // address and fixed fields.
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // 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()));
- // Check that the reserved values have been assigned.
- EXPECT_EQ("10.0.0.7", client.config_.siaddr_.toText());
- EXPECT_EQ("some-name.example.org", client.config_.sname_);
- EXPECT_EQ("bootfile.efi", client.config_.boot_file_name_);
- }
- // This test checks the following scenario:
- // 1. Client A performs 4-way exchange and obtains a lease from the dynamic pool.
- // 2. Reservation is created for the client A using an address out of the dynamic
- // pool.
- // 3. Client A renews the lease.
- // 4. Server responds with DHCPNAK to indicate that the client should stop using
- // an address for which it has a lease. Server doesn't want to renew an
- // address for which the client doesn't have a reservation, while it has
- // a reservation for a different address.
- // 5. Client A receives a DHCPNAK and returns to the DHCP server discovery.
- // 6. Client A performs a 4-way exchange with a server and the server allocates
- // a reserved address to the Client A.
- // 7. Client A renews the allocated address and the server returns a DHCPACK.
- // 8. Reservation for the Client A is removed.
- // 9. Client A renews the (previously reserved) lease and the server returns
- // DHCPNAK because the address in use is neither reserved nor belongs to
- // the dynamic pool.
- // 10. Client A returns to the DHCP server discovery.
- // 11. Client A uses 4-way exchange to obtain a lease from the dynamic pool.
- // 12. The new address that the Client A is using is reserved for Client B.
- // Client A still holds this address.
- // 13. Client B uses 4-way exchange to obtain a new lease.
- // 14. The server determines that the Client B has a reservation for the
- // address which is in use by Client A and offers an address different
- // than reserved.
- // 15. Client B requests the allocation of the offered address and the server
- // allocates this address.
- // 16. Client A renews the lease.
- // 17. The server determines that the address that Client A is using is reserved
- // for Client B. The server returns DHCPNAK to the Client A.
- // 18. Client B uses 4-way exchange to obtain the reserved lease but the lease
- // for the Client A hasn't been removed yet. Client B is assigned the same
- // address it has been using.
- // 19. Client A uses 4-way exchange to allocate a new lease.
- // 20. The server allocates a new lease from the dynamic pool but it avoids
- // allocating the address reserved for the Client B.
- // 21. Client B uses 4-way exchange to obtain a new lease.
- // 22. The server finally allocates a reserved address to the Client B.
- TEST_F(DORATest, reservationsWithConflicts) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Configure DHCP server.
- configure(DORA_CONFIGS[0], *client.getServer());
- // Client A performs 4-way exchange and obtains a lease from the
- // dynamic pool.
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("10.0.0.50"))));
- // 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()));
- // Make sure that the client has got the lease with the requested address.
- ASSERT_EQ("10.0.0.50", client.config_.lease_.addr_.toText());
- configure(DORA_CONFIGS[0], false);
- // Reservation is created for the client A using an address out of the
- // dynamic pool.
- HostPtr host(new Host(&client.getHWAddress()->hwaddr_[0],
- client.getHWAddress()->hwaddr_.size(),
- Host::IDENT_HWADDR, SubnetID(1),
- SubnetID(0), IOAddress("10.0.0.9")));
- CfgMgr::instance().getStagingCfg()->getCfgHosts()->add(host);
- CfgMgr::instance().commit();
- // Let's transition the client to Renewing state.
- client.setState(Dhcp4Client::RENEWING);
- // Set the unicast destination address to indicate that it is a renewal.
- client.setDestAddress(IOAddress("10.0.0.1"));
- ASSERT_NO_THROW(client.doRequest());
- // Client should get the DHCPNAK from the server because the client has
- // a reservation for a different address that it is trying to renew.
- resp = client.getContext().response_;
- ASSERT_EQ(DHCPNAK, static_cast<int>(resp->getType()));
- // A conforming client would go back to the server discovery.
- client.setState(Dhcp4Client::SELECTING);
- // Obtain a lease from the server using the 4-way exchange.
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPACK with a reserved
- // address
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- ASSERT_EQ("10.0.0.9", client.config_.lease_.addr_.toText());
- // Client A renews the allocated address.
- client.setState(Dhcp4Client::RENEWING);
- // Set the unicast destination address to indicate that it is a renewal.
- client.setDestAddress(IOAddress("10.0.0.1"));
- ASSERT_NO_THROW(client.doRequest());
- // Make sure the server responded and renewed the client's address.
- resp = client.getContext().response_;
- ASSERT_EQ("10.0.0.9", client.config_.lease_.addr_.toText());
- // By reconfiguring the server, we remove the existing reservations.
- configure(DORA_CONFIGS[0]);
- // Try to renew the existing lease again.
- ASSERT_NO_THROW(client.doRequest());
- // The reservation has been removed, so the server should respond with
- // a DHCPNAK because the address that the client is using doesn't belong
- // to a dynamic pool.
- resp = client.getContext().response_;
- ASSERT_EQ(DHCPNAK, static_cast<int>(resp->getType()));
- // A conforming client would go back to the server discovery.
- client.setState(Dhcp4Client::SELECTING);
- // Obtain a lease from the server using the 4-way exchange.
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Obtain the subnet to which the returned address belongs.
- Subnet4Ptr subnet = CfgMgr::instance().getCurrentCfg()->getCfgSubnets4()->
- selectSubnet(client.config_.lease_.addr_);
- ASSERT_TRUE(subnet);
- // Make sure that the address has been allocated from the dynamic pool.
- ASSERT_TRUE(subnet->inPool(Lease::TYPE_V4, client.config_.lease_.addr_));
- // Remember the address allocated in the dynamic pool.
- IOAddress in_pool_addr = client.config_.lease_.addr_;
- // Create Client B.
- Dhcp4Client clientB(client.getServer());
- clientB.modifyHWAddr();
- // Create reservation for the Client B, for the address that the
- // Client A is using.
- configure(DORA_CONFIGS[0], false);
- host.reset(new Host(&clientB.getHWAddress()->hwaddr_[0],
- clientB.getHWAddress()->hwaddr_.size(),
- Host::IDENT_HWADDR, SubnetID(1),
- SubnetID(0), in_pool_addr));
- CfgMgr::instance().getStagingCfg()->getCfgHosts()->add(host);
- CfgMgr::instance().commit();
- // Client B performs a DHCPDISCOVER.
- clientB.setState(Dhcp4Client::SELECTING);
- // The server determines that the address reserved for Client B is
- // in use by Client A so it offers a different address.
- ASSERT_NO_THROW(clientB.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- ASSERT_TRUE(clientB.getContext().response_);
- ASSERT_EQ(DHCPACK, static_cast<int>(clientB.getContext().response_->getType()));
- IOAddress client_b_addr = clientB.config_.lease_.addr_;
- ASSERT_NE(client_b_addr, in_pool_addr);
- // Client A renews the lease.
- client.setState(Dhcp4Client::RENEWING);
- // Set the unicast destination address to indicate that it is a renewal.
- client.setDestAddress(IOAddress(in_pool_addr));
- ASSERT_NO_THROW(client.doRequest());
- // Client A should get a DHCPNAK because it is using an address reserved
- // for Client B.
- resp = client.getContext().response_;
- ASSERT_EQ(DHCPNAK, static_cast<int>(resp->getType()));
- // Client B performs 4-way exchange but still gets an address from the
- // dynamic pool, because Client A hasn't obtained a new lease, so it is
- // still using an address reserved for Client B.
- clientB.setState(Dhcp4Client::SELECTING);
- // Obtain a lease from the server using the 4-way exchange.
- ASSERT_NO_THROW(clientB.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // Make sure that the server responded.
- ASSERT_TRUE(clientB.getContext().response_);
- ASSERT_EQ(DHCPACK, static_cast<int>(clientB.getContext().response_->getType()));
- ASSERT_NE(clientB.config_.lease_.addr_, in_pool_addr);
- ASSERT_EQ(client_b_addr, clientB.config_.lease_.addr_);
- // Client B renews its lease.
- clientB.setState(Dhcp4Client::RENEWING);
- clientB.setDestAddress(IOAddress("10.0.0.1"));
- ASSERT_NO_THROW(clientB.doRequest());
- // The server should renew the client's B lease because the address
- // reserved for client B is still in use by the client A.
- ASSERT_TRUE(clientB.getContext().response_);
- EXPECT_EQ(DHCPACK, static_cast<int>(clientB.getContext().response_->getType()));
- ASSERT_NE(clientB.config_.lease_.addr_, in_pool_addr);
- ASSERT_EQ(client_b_addr, clientB.config_.lease_.addr_);
- // Client A performs 4-way exchange.
- client.setState(Dhcp4Client::SELECTING);
- // Revert to the broadcast address for the selecting client.
- client.setDestAddress(IOAddress::IPV4_BCAST_ADDRESS());
- // Obtain a lease from the server using the 4-way exchange.
- ASSERT_NO_THROW(client.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // The server should have assigned a different address than the one
- // reserved for the Client B.
- ASSERT_NE(client.config_.lease_.addr_, in_pool_addr);
- subnet = CfgMgr::instance().getCurrentCfg()->getCfgSubnets4()->
- selectSubnet(client.config_.lease_.addr_);
- ASSERT_TRUE(subnet);
- ASSERT_TRUE(subnet->inPool(Lease::TYPE_V4, client.config_.lease_.addr_));
- // Client B renews again.
- ASSERT_NO_THROW(clientB.doRequest());
- // The client B should now receive the DHCPNAK from the server because
- // the reserved address is now available and the client should
- // revert to the DHCPDISCOVER to obtain it.
- ASSERT_TRUE(clientB.getContext().response_);
- EXPECT_EQ(DHCPNAK, static_cast<int>(clientB.getContext().response_->getType()));
- // Client B performs 4-way exchange and obtains a lease for the
- // reserved address.
- clientB.setState(Dhcp4Client::SELECTING);
- ASSERT_NO_THROW(clientB.doDORA(boost::shared_ptr<
- IOAddress>(new IOAddress("0.0.0.0"))));
- // Make sure that the server responded.
- ASSERT_TRUE(clientB.getContext().response_);
- resp = clientB.getContext().response_;
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- ASSERT_EQ(in_pool_addr, clientB.config_.lease_.addr_);
- }
- /// This test verifies that after a client completes its DORA exchange,
- /// appropriate statistics are updated.
- TEST_F(DORATest, statisticsDORA) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Configure DHCP server.
- configure(DORA_CONFIGS[0], *client.getServer());
- // Perform 4-way exchange with the server but to not request any
- // specific address in the DHCPDISCOVER message.
- ASSERT_NO_THROW(client.doDORA());
- // 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()));
- // Ok, let's check the statistics.
- using namespace isc::stats;
- StatsMgr& mgr = StatsMgr::instance();
- ObservationPtr pkt4_received = mgr.getObservation("pkt4-received");
- ObservationPtr pkt4_discover_received = mgr.getObservation("pkt4-discover-received");
- ObservationPtr pkt4_offer_sent = mgr.getObservation("pkt4-offer-sent");
- ObservationPtr pkt4_request_received = mgr.getObservation("pkt4-request-received");
- ObservationPtr pkt4_ack_sent = mgr.getObservation("pkt4-ack-sent");
- ObservationPtr pkt4_sent = mgr.getObservation("pkt4-sent");
- // All expected statistics must be present.
- ASSERT_TRUE(pkt4_received);
- ASSERT_TRUE(pkt4_discover_received);
- ASSERT_TRUE(pkt4_offer_sent);
- ASSERT_TRUE(pkt4_request_received);
- ASSERT_TRUE(pkt4_ack_sent);
- ASSERT_TRUE(pkt4_sent);
- // They also must have expected values.
- EXPECT_EQ(2, pkt4_received->getInteger().first);
- EXPECT_EQ(1, pkt4_discover_received->getInteger().first);
- EXPECT_EQ(1, pkt4_offer_sent->getInteger().first);
- EXPECT_EQ(1, pkt4_request_received->getInteger().first);
- EXPECT_EQ(1, pkt4_ack_sent->getInteger().first);
- EXPECT_EQ(2, pkt4_sent->getInteger().first);
- // Let the client send request 3 times, which should make the server
- // to send 3 acks.
- client.setState(Dhcp4Client::RENEWING);
- ASSERT_NO_THROW(client.doRequest());
- ASSERT_NO_THROW(client.doRequest());
- ASSERT_NO_THROW(client.doRequest());
- // Let's see if the stats are properly updated.
- EXPECT_EQ(5, pkt4_received->getInteger().first);
- EXPECT_EQ(1, pkt4_discover_received->getInteger().first);
- EXPECT_EQ(1, pkt4_offer_sent->getInteger().first);
- EXPECT_EQ(4, pkt4_request_received->getInteger().first);
- EXPECT_EQ(4, pkt4_ack_sent->getInteger().first);
- EXPECT_EQ(5, pkt4_sent->getInteger().first);
- }
- // This test verifies that after a client completes an exchange that result
- // in NAK, appropriate statistics are updated.
- TEST_F(DORATest, statisticsNAK) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Configure DHCP server.
- configure(DORA_CONFIGS[0], *client.getServer());
- // Obtain a lease from the server using the 4-way exchange.
- // Get a lease.
- client.doDORA();
- // Wipe all stats.
- isc::stats::StatsMgr::instance().removeAll();
- client.setState(Dhcp4Client::INIT_REBOOT);
- client.config_.lease_.addr_ = IOAddress("10.0.0.30");
- ASSERT_NO_THROW(client.doRequest());
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- Pkt4Ptr resp = client.getContext().response_;
- EXPECT_EQ(DHCPNAK, static_cast<int>(resp->getType()));
- using namespace isc::stats;
- StatsMgr& mgr = StatsMgr::instance();
- ObservationPtr pkt4_received = mgr.getObservation("pkt4-received");
- ObservationPtr pkt4_request_received = mgr.getObservation("pkt4-request-received");
- ObservationPtr pkt4_ack_sent = mgr.getObservation("pkt4-ack-sent");
- ObservationPtr pkt4_nak_sent = mgr.getObservation("pkt4-nak-sent");
- ObservationPtr pkt4_sent = mgr.getObservation("pkt4-sent");
- // All expected statistics must be present.
- ASSERT_TRUE(pkt4_received);
- ASSERT_TRUE(pkt4_request_received);
- ASSERT_FALSE(pkt4_ack_sent); // No acks were sent, no such statistic expected.
- ASSERT_TRUE(pkt4_nak_sent);
- ASSERT_TRUE(pkt4_sent);
- // They also must have expected values.
- EXPECT_EQ(1, pkt4_received->getInteger().first);
- EXPECT_EQ(1, pkt4_request_received->getInteger().first);
- EXPECT_EQ(1, pkt4_nak_sent->getInteger().first);
- EXPECT_EQ(1, pkt4_sent->getInteger().first);
- }
- void
- DORATest::testMultiStageBoot(const unsigned int config_index) {
- Dhcp4Client client(Dhcp4Client::SELECTING);
- // Configure DHCP server.
- ASSERT_NO_THROW(configure(DORA_CONFIGS[config_index],
- *client.getServer()));
- // Stage 1: get the first lease for our client. In PXE boot, it would be
- // a stage when the BIOS requests a lease.
- // Include client id apart from the MAC address.
- client.includeClientId("10:21:32:AB:CD:EF");
- ASSERT_NO_THROW(client.doDORA());
- // 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()));
- // Make sure that the client has got the lease which belongs
- // to a pool.
- IOAddress leased_address1 = client.config_.lease_.addr_;
- Subnet4Ptr subnet = CfgMgr::instance().getCurrentCfg()->getCfgSubnets4()->
- selectSubnet(leased_address1);
- ASSERT_TRUE(subnet);
- // Make sure that the address has been allocated from the dynamic pool.
- ASSERT_TRUE(subnet->inPool(Lease::TYPE_V4, leased_address1));
- // Stage 2: the client with a given MAC address has a lease in the
- // lease database. The installer comes up and uses the same MAC address
- // but generates a different client id. The server should treat the
- // client with modified client identifier as a different client and
- // create a new lease for it.
- // Modify client identifier.
- client.includeClientId("11:54:45:AB:AA:FE");
- ASSERT_NO_THROW(client.doDORA());
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Make sure that the client has got the lease which belongs
- // to a pool.
- IOAddress leased_address2 = client.config_.lease_.addr_;
- // Make sure that the address has been allocated from the dynamic pool.
- ASSERT_TRUE(subnet->inPool(Lease::TYPE_V4, leased_address2));
- // The client should have got a new lease.
- ASSERT_NE(leased_address1, leased_address2);
- // Modify client identifier again.
- client.includeClientId("22:34:AC:BE:44:54");
- ASSERT_NO_THROW(client.doDORA());
- // Make sure that the server responded.
- ASSERT_TRUE(client.getContext().response_);
- resp = client.getContext().response_;
- // Make sure that the server has responded with DHCPACK.
- ASSERT_EQ(DHCPACK, static_cast<int>(resp->getType()));
- // Make sure that the client has got the lease which belongs
- // to a pool.
- IOAddress leased_address3 = client.config_.lease_.addr_;
- // Make sure that the address has been allocated from the dynamic pool.
- ASSERT_TRUE(subnet->inPool(Lease::TYPE_V4, leased_address3));
- // The client should have got a new lease.
- ASSERT_NE(leased_address1, leased_address3);
- ASSERT_NE(leased_address2, leased_address3);
- }
- // Test that the client using the same hardware address but multiple
- // client identifiers will obtain multiple leases.
- TEST_F(DORATest, multiStageBoot) {
- // DORA_CONFIGS[0] to be used for server configuration.
- testMultiStageBoot(0);
- }
- // Starting tests which require MySQL backend availability. Those tests
- // will not be executed if Kea has been compiled without the
- // --with-dhcp-mysql.
- #ifdef HAVE_MYSQL
- /// @brief Test fixture class for the test utilizing MySQL database backend.
- class DORAMySQLTest : public DORATest {
- public:
- /// @brief Constructor.
- ///
- /// Recreates MySQL schema for a test.
- DORAMySQLTest() : DORATest() {
- destroyMySQLSchema();
- createMySQLSchema();
- }
- /// @brief Destructor.
- ///
- /// Destroys MySQL schema.
- virtual ~DORAMySQLTest() {
- destroyMySQLSchema();
- }
- };
- // Test that the client using the same hardware address but multiple
- // client identifiers will obtain multiple leases (MySQL lease database).
- TEST_F(DORAMySQLTest, multiStageBoot) {
- // DORA_CONFIGS[7] to be used for server configuration.
- testMultiStageBoot(7);
- }
- #endif
- // Starting tests which require MySQL backend availability. Those tests
- // will not be executed if Kea has been compiled without the
- // --with-dhcp-pgsql.
- #ifdef HAVE_PGSQL
- /// @brief Test fixture class for the test utilizing PostgreSQL database backend.
- class DORAPgSQLTest : public DORATest {
- public:
- /// @brief Constructor.
- ///
- /// Recreates PgSQL schema for a test.
- DORAPgSQLTest() : DORATest() {
- destroyPgSQLSchema();
- createPgSQLSchema();
- }
- /// @brief Destructor.
- ///
- /// Destroys PgSQL schema.
- virtual ~DORAPgSQLTest() {
- destroyPgSQLSchema();
- }
- };
- // Test that the client using the same hardware address but multiple
- // client identifiers will obtain multiple leases (PostgreSQL lease database).
- TEST_F(DORAPgSQLTest, multiStageBoot) {
- // DORA_CONFIGS[8] to be used for server configuration.
- testMultiStageBoot(8);
- }
- #endif
- #ifdef HAVE_CQL
- /// @brief Test fixture class for the test utilizing Cassandra database backend.
- class DORACQLTest : public DORATest {
- public:
- /// @brief Constructor.
- ///
- /// Recreates CQL schema for a test.
- DORACQLTest() : DORATest() {
- destroyCqlSchema(false, true);
- createCqlSchema(false, true);
- }
- /// @brief Destructor.
- ///
- /// Destroys CQL schema.
- virtual ~DORACQLTest() {
- destroyCqlSchema(false, true);
- }
- };
- // Test that the client using the same hardware address but multiple
- // client identifiers will obtain multiple leases (CQL lease database).
- TEST_F(DORACQLTest, multiStageBoot) {
- // DORA_CONFIGS[9] to be used for server configuration.
- testMultiStageBoot(9);
- }
- #endif
- } // end of anonymous namespace
|