option_unittest.cc 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605
  1. // Copyright (C) 2011-2016 Internet Systems Consortium, Inc. ("ISC")
  2. //
  3. // This Source Code Form is subject to the terms of the Mozilla Public
  4. // License, v. 2.0. If a copy of the MPL was not distributed with this
  5. // file, You can obtain one at http://mozilla.org/MPL/2.0/.
  6. #include <config.h>
  7. #include <dhcp/dhcp6.h>
  8. #include <dhcp/libdhcp++.h>
  9. #include <dhcp/option.h>
  10. #include <exceptions/exceptions.h>
  11. #include <util/buffer.h>
  12. #include <boost/bind.hpp>
  13. #include <boost/shared_ptr.hpp>
  14. #include <boost/scoped_ptr.hpp>
  15. #include <gtest/gtest.h>
  16. #include <iostream>
  17. #include <sstream>
  18. #include <arpa/inet.h>
  19. using namespace std;
  20. using namespace isc;
  21. using namespace isc::dhcp;
  22. using namespace isc::util;
  23. using boost::scoped_ptr;
  24. namespace {
  25. /// @brief A class which derives from option and exposes protected members.
  26. class NakedOption : public Option {
  27. public:
  28. /// @brief Constructor
  29. ///
  30. /// Sets the universe and option type to arbitrary test values.
  31. NakedOption() : Option(Option::V6, 258) {
  32. }
  33. using Option::unpackOptions;
  34. };
  35. class OptionTest : public ::testing::Test {
  36. public:
  37. OptionTest(): buf_(255), outBuf_(255) {
  38. for (unsigned i = 0; i < 255; i++) {
  39. buf_[i] = 255 - i;
  40. }
  41. }
  42. OptionBuffer buf_;
  43. OutputBuffer outBuf_;
  44. };
  45. // Basic tests for V4 functionality
  46. TEST_F(OptionTest, v4_basic) {
  47. scoped_ptr<Option> opt;
  48. EXPECT_NO_THROW(opt.reset(new Option(Option::V4, 17)));
  49. EXPECT_EQ(Option::V4, opt->getUniverse());
  50. EXPECT_EQ(17, opt->getType());
  51. EXPECT_EQ(0, opt->getData().size());
  52. EXPECT_EQ(2, opt->len()); // just v4 header
  53. EXPECT_NO_THROW(opt.reset());
  54. // V4 options have type 0...255
  55. EXPECT_THROW(opt.reset(new Option(Option::V4, 256)), BadValue);
  56. // 0 is a special PAD option
  57. EXPECT_THROW(opt.reset(new Option(Option::V4, 0)), BadValue);
  58. // 255 is a special END option
  59. EXPECT_THROW(opt.reset(new Option(Option::V4, 255)), BadValue);
  60. }
  61. const uint8_t dummyPayload[] =
  62. { 1, 2, 3, 4};
  63. TEST_F(OptionTest, v4_data1) {
  64. vector<uint8_t> data(dummyPayload, dummyPayload + sizeof(dummyPayload));
  65. scoped_ptr<Option> opt;
  66. // Create DHCPv4 option of type 123 that contains 4 bytes of data.
  67. ASSERT_NO_THROW(opt.reset(new Option(Option::V4, 123, data)));
  68. // Check that content is reported properly
  69. EXPECT_EQ(123, opt->getType());
  70. vector<uint8_t> optData = opt->getData();
  71. ASSERT_EQ(optData.size(), data.size());
  72. EXPECT_TRUE(optData == data);
  73. EXPECT_EQ(2, opt->getHeaderLen());
  74. EXPECT_EQ(6, opt->len());
  75. // Now store that option into a buffer
  76. OutputBuffer buf(100);
  77. EXPECT_NO_THROW(opt->pack(buf));
  78. // Check content of that buffer:
  79. // 2 byte header + 4 bytes data
  80. ASSERT_EQ(6, buf.getLength());
  81. // That's how this option is supposed to look like
  82. uint8_t exp[] = { 123, 4, 1, 2, 3, 4 };
  83. /// TODO: use vector<uint8_t> getData() when it will be implemented
  84. EXPECT_EQ(0, memcmp(exp, buf.getData(), 6));
  85. // Check that we can destroy that option
  86. EXPECT_NO_THROW(opt.reset());
  87. }
  88. // This is almost the same test as v4_data1, but it uses a different
  89. // constructor
  90. TEST_F(OptionTest, v4_data2) {
  91. vector<uint8_t> data(dummyPayload, dummyPayload + sizeof(dummyPayload));
  92. vector<uint8_t> expData = data;
  93. // Add fake data in front and end. Main purpose of this test is to check
  94. // that only subset of the whole vector can be used for creating option.
  95. data.insert(data.begin(), 56);
  96. data.push_back(67);
  97. // Data contains extra garbage at beginning and at the end. It should be
  98. // ignored, as we pass interators to proper data. Only subset (limited by
  99. // iterators) of the vector should be used.
  100. // expData contains expected content (just valid data, without garbage).
  101. scoped_ptr<Option> opt;
  102. // Create DHCPv4 option of type 123 that contains
  103. // 4 bytes (sizeof(dummyPayload).
  104. ASSERT_NO_THROW(
  105. opt.reset(new Option(Option::V4, 123, data.begin() + 1,
  106. data.end() - 1));
  107. );
  108. // Check that content is reported properly
  109. EXPECT_EQ(123, opt->getType());
  110. vector<uint8_t> optData = opt->getData();
  111. ASSERT_EQ(optData.size(), expData.size());
  112. EXPECT_TRUE(optData == expData);
  113. EXPECT_EQ(2, opt->getHeaderLen());
  114. EXPECT_EQ(6, opt->len());
  115. // Now store that option into a buffer
  116. OutputBuffer buf(100);
  117. EXPECT_NO_THROW(opt->pack(buf));
  118. // Check content of that buffer
  119. // 2 byte header + 4 bytes data
  120. ASSERT_EQ(6, buf.getLength());
  121. // That's how this option is supposed to look like
  122. uint8_t exp[] = { 123, 4, 1, 2, 3, 4 };
  123. /// TODO: use vector<uint8_t> getData() when it will be implemented
  124. EXPECT_EQ(0, memcmp(exp, buf.getData(), 6));
  125. // Check that we can destroy that option
  126. EXPECT_NO_THROW(opt.reset());
  127. }
  128. TEST_F(OptionTest, v4_toText) {
  129. vector<uint8_t> buf(3);
  130. buf[0] = 0;
  131. buf[1] = 0xf;
  132. buf[2] = 0xff;
  133. Option opt(Option::V4, 253, buf);
  134. EXPECT_EQ("type=253, len=003: 00:0f:ff", opt.toText());
  135. }
  136. // Test converting option to the hexadecimal representation.
  137. TEST_F(OptionTest, v4_toHexString) {
  138. std::vector<uint8_t> payload;
  139. for (unsigned int i = 0; i < 16; ++i) {
  140. payload.push_back(static_cast<uint8_t>(i));
  141. }
  142. Option opt(Option::V4, 122, payload);
  143. EXPECT_EQ("0x000102030405060708090A0B0C0D0E0F", opt.toHexString());
  144. EXPECT_EQ("0x7A10000102030405060708090A0B0C0D0E0F",
  145. opt.toHexString(true));
  146. // Test empty option.
  147. Option opt_empty(Option::V4, 65, std::vector<uint8_t>());
  148. EXPECT_TRUE(opt_empty.toHexString().empty());
  149. EXPECT_EQ("0x4100", opt_empty.toHexString(true));
  150. // Test too long option. We can't simply create such option by
  151. // providing a long payload, because class constructor would not
  152. // accept it. Instead we'll add two long sub options after we
  153. // create an option instance.
  154. Option opt_too_long(Option::V4, 33);
  155. // Both suboptions have payloads of 150 bytes.
  156. std::vector<uint8_t> long_payload(150, 1);
  157. OptionPtr sub1(new Option(Option::V4, 100, long_payload));
  158. OptionPtr sub2(new Option(Option::V4, 101, long_payload));
  159. opt_too_long.addOption(sub1);
  160. opt_too_long.addOption(sub2);
  161. // The toHexString() should throw exception.
  162. EXPECT_THROW(opt_too_long.toHexString(), isc::OutOfRange);
  163. }
  164. // Tests simple constructor
  165. TEST_F(OptionTest, v6_basic) {
  166. scoped_ptr<Option> opt(new Option(Option::V6, 1));
  167. EXPECT_EQ(Option::V6, opt->getUniverse());
  168. EXPECT_EQ(1, opt->getType());
  169. EXPECT_EQ(0, opt->getData().size());
  170. EXPECT_EQ(4, opt->len()); // Just v6 header
  171. EXPECT_NO_THROW(opt.reset());
  172. }
  173. // Tests constructor used in packet reception. Option contains actual data
  174. TEST_F(OptionTest, v6_data1) {
  175. for (unsigned i = 0; i < 32; i++) {
  176. buf_[i] = 100 + i;
  177. }
  178. // Create option with seven bytes of data.
  179. scoped_ptr<Option> opt(new Option(Option::V6, 333, // Type
  180. buf_.begin() + 3, // Begin offset
  181. buf_.begin() + 10)); // End offset
  182. EXPECT_EQ(333, opt->getType());
  183. ASSERT_EQ(11, opt->len());
  184. ASSERT_EQ(7, opt->getData().size());
  185. EXPECT_EQ(0, memcmp(&buf_[3], &opt->getData()[0], 7) );
  186. opt->pack(outBuf_);
  187. EXPECT_EQ(11, outBuf_.getLength());
  188. const uint8_t* out = static_cast<const uint8_t*>(outBuf_.getData());
  189. EXPECT_EQ(out[0], 333 / 256); // Type
  190. EXPECT_EQ(out[1], 333 % 256);
  191. EXPECT_EQ(out[2], 0); // Length
  192. EXPECT_EQ(out[3], 7);
  193. // Payload
  194. EXPECT_EQ(0, memcmp(&buf_[3], out + 4, 7));
  195. EXPECT_NO_THROW(opt.reset());
  196. }
  197. // Another test that tests the same thing, just with different input parameters.
  198. TEST_F(OptionTest, v6_data2) {
  199. buf_[0] = 0xa1;
  200. buf_[1] = 0xa2;
  201. buf_[2] = 0xa3;
  202. buf_[3] = 0xa4;
  203. // Create an option (unpack content)
  204. scoped_ptr<Option> opt(new Option(Option::V6, D6O_CLIENTID,
  205. buf_.begin(), buf_.begin() + 4));
  206. // Pack this option
  207. opt->pack(outBuf_);
  208. // 4 bytes header + 4 bytes content
  209. EXPECT_EQ(8, opt->len());
  210. EXPECT_EQ(D6O_CLIENTID, opt->getType());
  211. EXPECT_EQ(8, outBuf_.getLength());
  212. // Check if pack worked properly:
  213. // If option type is correct
  214. const uint8_t* out = static_cast<const uint8_t*>(outBuf_.getData());
  215. EXPECT_EQ(D6O_CLIENTID, out[0] * 256 + out[1]);
  216. // If option length is correct
  217. EXPECT_EQ(4, out[2] * 256 + out[3]);
  218. // If option content is correct
  219. EXPECT_EQ(0, memcmp(&buf_[0], out + 4, 4));
  220. EXPECT_NO_THROW(opt.reset());
  221. }
  222. // Check that an option can contain 2 suboptions:
  223. // opt1
  224. // +----opt2
  225. // |
  226. // +----opt3
  227. //
  228. TEST_F(OptionTest, v6_suboptions1) {
  229. for (unsigned i = 0; i < 128; i++) {
  230. buf_[i] = 100 + i;
  231. }
  232. scoped_ptr<Option> opt1(new Option(Option::V6, 65535, // Type
  233. buf_.begin(), // 3 bytes of data
  234. buf_.begin() + 3));
  235. OptionPtr opt2(new Option(Option::V6, 13));
  236. OptionPtr opt3(new Option(Option::V6, 7,
  237. buf_.begin() + 3,
  238. buf_.begin() + 8)); // 5 bytes of data
  239. opt1->addOption(opt2);
  240. opt1->addOption(opt3);
  241. // opt2 len = 4 (just header)
  242. // opt3 len = 9 4(header)+5(data)
  243. // opt1 len = 7 + suboptions() = 7 + 4 + 9 = 20
  244. EXPECT_EQ(4, opt2->len());
  245. EXPECT_EQ(9, opt3->len());
  246. EXPECT_EQ(20, opt1->len());
  247. uint8_t expected[] = {
  248. 0xff, 0xff, 0, 16, 100, 101, 102,
  249. 0, 7, 0, 5, 103, 104, 105, 106, 107,
  250. 0, 13, 0, 0 // no data at all
  251. };
  252. opt1->pack(outBuf_);
  253. EXPECT_EQ(20, outBuf_.getLength());
  254. // Payload
  255. EXPECT_EQ(0, memcmp(outBuf_.getData(), expected, 20) );
  256. EXPECT_NO_THROW(opt1.reset());
  257. }
  258. // Check that an option can contain nested suboptions:
  259. // opt1
  260. // +----opt2
  261. // |
  262. // +----opt3
  263. //
  264. TEST_F(OptionTest, v6_suboptions2) {
  265. for (unsigned i = 0; i < 128; i++) {
  266. buf_[i] = 100 + i;
  267. }
  268. scoped_ptr<Option> opt1(new Option(Option::V6, 65535, // Type
  269. buf_.begin(), buf_.begin() + 3));
  270. OptionPtr opt2(new Option(Option::V6, 13));
  271. OptionPtr opt3(new Option(Option::V6, 7,
  272. buf_.begin() + 3,
  273. buf_.begin() + 8));
  274. opt1->addOption(opt2);
  275. opt2->addOption(opt3);
  276. // opt3 len = 9 4(header)+5(data)
  277. // opt2 len = 4 (just header) + len(opt3)
  278. // opt1 len = 7 + len(opt2)
  279. uint8_t expected[] = {
  280. 0xff, 0xff, 0, 16, 100, 101, 102,
  281. 0, 13, 0, 9,
  282. 0, 7, 0, 5, 103, 104, 105, 106, 107,
  283. };
  284. opt1->pack(outBuf_);
  285. EXPECT_EQ(20, outBuf_.getLength());
  286. // Payload
  287. EXPECT_EQ(0, memcmp(outBuf_.getData(), expected, 20) );
  288. EXPECT_NO_THROW(opt1.reset());
  289. }
  290. TEST_F(OptionTest, v6_addgetdel) {
  291. for (unsigned i = 0; i < 128; i++) {
  292. buf_[i] = 100 + i;
  293. }
  294. scoped_ptr<Option> parent(new Option(Option::V6, 65535)); // Type
  295. OptionPtr opt1(new Option(Option::V6, 1));
  296. OptionPtr opt2(new Option(Option::V6, 2));
  297. OptionPtr opt3(new Option(Option::V6, 2));
  298. parent->addOption(opt1);
  299. parent->addOption(opt2);
  300. // getOption() test
  301. EXPECT_EQ(opt1, parent->getOption(1));
  302. EXPECT_EQ(opt2, parent->getOption(2));
  303. // Expect NULL
  304. EXPECT_EQ(OptionPtr(), parent->getOption(4));
  305. // Now there are 2 options of type 2
  306. parent->addOption(opt3);
  307. // Let's delete one of them
  308. EXPECT_EQ(true, parent->delOption(2));
  309. // There still should be the other option 2
  310. EXPECT_NE(OptionPtr(), parent->getOption(2));
  311. // Let's delete the other option 2
  312. EXPECT_EQ(true, parent->delOption(2));
  313. // No more options with type=2
  314. EXPECT_EQ(OptionPtr(), parent->getOption(2));
  315. // Let's try to delete - should fail
  316. EXPECT_TRUE(false == parent->delOption(2));
  317. }
  318. TEST_F(OptionTest, v6_toText) {
  319. buf_[0] = 0;
  320. buf_[1] = 0xf;
  321. buf_[2] = 0xff;
  322. OptionPtr opt(new Option(Option::V6, 258, buf_.begin(), buf_.begin() + 3 ));
  323. EXPECT_EQ("type=00258, len=00003: 00:0f:ff", opt->toText());
  324. }
  325. // Test converting option to the hexadecimal representation.
  326. TEST_F(OptionTest, v6_toHexString) {
  327. std::vector<uint8_t> payload;
  328. for (unsigned int i = 0; i < 16; ++i) {
  329. payload.push_back(static_cast<uint8_t>(i));
  330. }
  331. Option opt(Option::V6, 12202, payload);
  332. EXPECT_EQ("0x000102030405060708090A0B0C0D0E0F", opt.toHexString());
  333. EXPECT_EQ("0x2FAA0010000102030405060708090A0B0C0D0E0F",
  334. opt.toHexString(true));
  335. // Test empty option.
  336. Option opt_empty(Option::V6, 65000, std::vector<uint8_t>());
  337. EXPECT_TRUE(opt_empty.toHexString().empty());
  338. EXPECT_EQ("0xFDE80000", opt_empty.toHexString(true));
  339. }
  340. TEST_F(OptionTest, getUintX) {
  341. buf_[0] = 0x5;
  342. buf_[1] = 0x4;
  343. buf_[2] = 0x3;
  344. buf_[3] = 0x2;
  345. buf_[4] = 0x1;
  346. // Five options with varying lengths
  347. OptionPtr opt1(new Option(Option::V6, 258, buf_.begin(), buf_.begin() + 1));
  348. OptionPtr opt2(new Option(Option::V6, 258, buf_.begin(), buf_.begin() + 2));
  349. OptionPtr opt3(new Option(Option::V6, 258, buf_.begin(), buf_.begin() + 3));
  350. OptionPtr opt4(new Option(Option::V6, 258, buf_.begin(), buf_.begin() + 4));
  351. OptionPtr opt5(new Option(Option::V6, 258, buf_.begin(), buf_.begin() + 5));
  352. EXPECT_EQ(5, opt1->getUint8());
  353. EXPECT_THROW(opt1->getUint16(), OutOfRange);
  354. EXPECT_THROW(opt1->getUint32(), OutOfRange);
  355. EXPECT_EQ(5, opt2->getUint8());
  356. EXPECT_EQ(0x0504, opt2->getUint16());
  357. EXPECT_THROW(opt2->getUint32(), OutOfRange);
  358. EXPECT_EQ(5, opt3->getUint8());
  359. EXPECT_EQ(0x0504, opt3->getUint16());
  360. EXPECT_THROW(opt3->getUint32(), OutOfRange);
  361. EXPECT_EQ(5, opt4->getUint8());
  362. EXPECT_EQ(0x0504, opt4->getUint16());
  363. EXPECT_EQ(0x05040302, opt4->getUint32());
  364. // The same as for 4-byte long, just get first 1,2 or 4 bytes
  365. EXPECT_EQ(5, opt5->getUint8());
  366. EXPECT_EQ(0x0504, opt5->getUint16());
  367. EXPECT_EQ(0x05040302, opt5->getUint32());
  368. }
  369. TEST_F(OptionTest, setUintX) {
  370. OptionPtr opt1(new Option(Option::V4, 125));
  371. OptionPtr opt2(new Option(Option::V4, 125));
  372. OptionPtr opt4(new Option(Option::V4, 125));
  373. // Verify setUint8
  374. opt1->setUint8(255);
  375. EXPECT_EQ(255, opt1->getUint8());
  376. opt1->pack(outBuf_);
  377. EXPECT_EQ(3, opt1->len());
  378. EXPECT_EQ(3, outBuf_.getLength());
  379. uint8_t exp1[] = {125, 1, 255};
  380. EXPECT_TRUE(0 == memcmp(exp1, outBuf_.getData(), 3));
  381. // Verify getUint16
  382. outBuf_.clear();
  383. opt2->setUint16(12345);
  384. opt2->pack(outBuf_);
  385. EXPECT_EQ(12345, opt2->getUint16());
  386. EXPECT_EQ(4, opt2->len());
  387. EXPECT_EQ(4, outBuf_.getLength());
  388. uint8_t exp2[] = {125, 2, 12345/256, 12345%256};
  389. EXPECT_TRUE(0 == memcmp(exp2, outBuf_.getData(), 4));
  390. // Verify getUint32
  391. outBuf_.clear();
  392. opt4->setUint32(0x12345678);
  393. opt4->pack(outBuf_);
  394. EXPECT_EQ(0x12345678, opt4->getUint32());
  395. EXPECT_EQ(6, opt4->len());
  396. EXPECT_EQ(6, outBuf_.getLength());
  397. uint8_t exp4[] = {125, 4, 0x12, 0x34, 0x56, 0x78};
  398. EXPECT_TRUE(0 == memcmp(exp4, outBuf_.getData(), 6));
  399. }
  400. TEST_F(OptionTest, setData) {
  401. // Verify data override with new buffer larger than initial option buffer
  402. // size.
  403. OptionPtr opt1(new Option(Option::V4, 125,
  404. buf_.begin(), buf_.begin() + 10));
  405. buf_.resize(20, 1);
  406. opt1->setData(buf_.begin(), buf_.end());
  407. opt1->pack(outBuf_);
  408. ASSERT_EQ(outBuf_.getLength() - opt1->getHeaderLen(), buf_.size());
  409. const uint8_t* test_data = static_cast<const uint8_t*>(outBuf_.getData());
  410. EXPECT_TRUE(0 == memcmp(&buf_[0], test_data + opt1->getHeaderLen(),
  411. buf_.size()));
  412. // Verify data override with new buffer shorter than initial option buffer
  413. // size.
  414. OptionPtr opt2(new Option(Option::V4, 125,
  415. buf_.begin(), buf_.begin() + 10));
  416. outBuf_.clear();
  417. buf_.resize(5, 1);
  418. opt2->setData(buf_.begin(), buf_.end());
  419. opt2->pack(outBuf_);
  420. ASSERT_EQ(outBuf_.getLength() - opt1->getHeaderLen(), buf_.size());
  421. test_data = static_cast<const uint8_t*>(outBuf_.getData());
  422. EXPECT_TRUE(0 == memcmp(&buf_[0], test_data + opt1->getHeaderLen(),
  423. buf_.size()));
  424. }
  425. // This test verifies that options can be compared using equals(OptionPtr)
  426. // method.
  427. TEST_F(OptionTest, equalsWithPointers) {
  428. // Five options with varying lengths
  429. OptionPtr opt1(new Option(Option::V6, 258, buf_.begin(), buf_.begin() + 1));
  430. OptionPtr opt2(new Option(Option::V6, 258, buf_.begin(), buf_.begin() + 2));
  431. OptionPtr opt3(new Option(Option::V6, 258, buf_.begin(), buf_.begin() + 3));
  432. // The same content as opt2, but different type
  433. OptionPtr opt4(new Option(Option::V6, 1, buf_.begin(), buf_.begin() + 2));
  434. // Another instance with the same type and content as opt2
  435. OptionPtr opt5(new Option(Option::V6, 258, buf_.begin(), buf_.begin() + 2));
  436. EXPECT_TRUE(opt1->equals(opt1));
  437. EXPECT_FALSE(opt1->equals(opt2));
  438. EXPECT_FALSE(opt1->equals(opt3));
  439. EXPECT_FALSE(opt1->equals(opt4));
  440. EXPECT_TRUE(opt2->equals(opt5));
  441. }
  442. // This test verifies that options can be compared using equals(Option) method.
  443. TEST_F(OptionTest, equals) {
  444. // Five options with varying lengths
  445. Option opt1(Option::V6, 258, buf_.begin(), buf_.begin() + 1);
  446. Option opt2(Option::V6, 258, buf_.begin(), buf_.begin() + 2);
  447. Option opt3(Option::V6, 258, buf_.begin(), buf_.begin() + 3);
  448. // The same content as opt2, but different type
  449. Option opt4(Option::V6, 1, buf_.begin(), buf_.begin() + 2);
  450. // Another instance with the same type and content as opt2
  451. Option opt5(Option::V6, 258, buf_.begin(), buf_.begin() + 2);
  452. EXPECT_TRUE(opt1.equals(opt1));
  453. EXPECT_FALSE(opt1.equals(opt2));
  454. EXPECT_FALSE(opt1.equals(opt3));
  455. EXPECT_FALSE(opt1.equals(opt4));
  456. EXPECT_TRUE(opt2.equals(opt5));
  457. }
  458. // This test verifies that the name of the option space being encapsulated by
  459. // the particular option can be set.
  460. TEST_F(OptionTest, setEncapsulatedSpace) {
  461. Option optv6(Option::V6, 258);
  462. EXPECT_TRUE(optv6.getEncapsulatedSpace().empty());
  463. optv6.setEncapsulatedSpace("dhcp6");
  464. EXPECT_EQ("dhcp6", optv6.getEncapsulatedSpace());
  465. Option optv4(Option::V4, 125);
  466. EXPECT_TRUE(optv4.getEncapsulatedSpace().empty());
  467. optv4.setEncapsulatedSpace("dhcp4");
  468. EXPECT_EQ("dhcp4", optv4.getEncapsulatedSpace());
  469. }
  470. }