message_unittest.cc 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  1. // Copyright (C) 2010 Internet Systems Consortium, Inc. ("ISC")
  2. //
  3. // Permission to use, copy, modify, and/or distribute this software for any
  4. // purpose with or without fee is hereby granted, provided that the above
  5. // copyright notice and this permission notice appear in all copies.
  6. //
  7. // THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
  8. // REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
  9. // AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
  10. // INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
  11. // LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
  12. // OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  13. // PERFORMANCE OF THIS SOFTWARE.
  14. // $Id$
  15. #include <exceptions/exceptions.h>
  16. #include <dns/buffer.h>
  17. #include <dns/exceptions.h>
  18. #include <dns/message.h>
  19. #include <dns/messagerenderer.h>
  20. #include <dns/question.h>
  21. #include <dns/rdataclass.h>
  22. #include <dns/rrclass.h>
  23. #include <dns/rrttl.h>
  24. #include <dns/rrtype.h>
  25. #include <gtest/gtest.h>
  26. #include <dns/tests/unittest_util.h>
  27. using isc::UnitTestUtil;
  28. using namespace std;
  29. using namespace isc::dns;
  30. using namespace isc::dns::rdata;
  31. //
  32. // Note: we need more tests, including:
  33. // parsing malformed headers
  34. // more complete tests about parsing/rendering header flags, opcode, rcode, etc.
  35. // tests for adding RRsets
  36. // tests for RRset/Question iterators
  37. // But, we'll ship with the current set of tests for now, partly because many
  38. // of the above are covered as part of other tests, and partly due to time
  39. // limitation. We also expect to revisit the fundamental design of the Message
  40. // class, at which point we'll also revise the tests including more cases.
  41. //
  42. const uint16_t Message::DEFAULT_MAX_UDPSIZE;
  43. namespace {
  44. class MessageTest : public ::testing::Test {
  45. protected:
  46. MessageTest() : obuffer(0), renderer(obuffer),
  47. message_parse(Message::PARSE),
  48. message_render(Message::RENDER)
  49. {}
  50. static Question factoryFromFile(const char* datafile);
  51. OutputBuffer obuffer;
  52. MessageRenderer renderer;
  53. Message message_parse;
  54. Message message_render;
  55. static void factoryFromFile(Message& message, const char* datafile);
  56. };
  57. const Name test_name("test.example.com");
  58. void
  59. MessageTest::factoryFromFile(Message& message, const char* datafile) {
  60. std::vector<unsigned char> data;
  61. UnitTestUtil::readWireData(datafile, data);
  62. InputBuffer buffer(&data[0], data.size());
  63. message.fromWire(buffer);
  64. }
  65. TEST_F(MessageTest, RcodeConstruct) {
  66. // normal cases
  67. EXPECT_EQ(0, Rcode(0).getCode());
  68. EXPECT_EQ(0xfff, Rcode(0xfff).getCode()); // possible max code
  69. // should fail on attempt of construction with an out of range code
  70. EXPECT_THROW(Rcode(0x1000), isc::OutOfRange);
  71. EXPECT_THROW(Rcode(0xffff), isc::OutOfRange);
  72. }
  73. TEST_F(MessageTest, RcodeToText) {
  74. EXPECT_EQ("NOERROR", Rcode::NOERROR().toText());
  75. EXPECT_EQ("BADVERS", Rcode::BADVERS().toText());
  76. EXPECT_EQ("17", Rcode(Rcode::BADVERS().getCode() + 1).toText());
  77. EXPECT_EQ("4095", Rcode(Rcode(0xfff)).toText());
  78. }
  79. TEST_F(MessageTest, fromWire) {
  80. factoryFromFile(message_parse, "message_fromWire1");
  81. EXPECT_EQ(0x1035, message_parse.getQid());
  82. EXPECT_EQ(Opcode::QUERY(), message_parse.getOpcode());
  83. EXPECT_EQ(Rcode::NOERROR(), message_parse.getRcode());
  84. EXPECT_TRUE(message_parse.getHeaderFlag(MessageFlag::QR()));
  85. EXPECT_TRUE(message_parse.getHeaderFlag(MessageFlag::RD()));
  86. EXPECT_TRUE(message_parse.getHeaderFlag(MessageFlag::AA()));
  87. QuestionPtr q = *message_parse.beginQuestion();
  88. EXPECT_EQ(test_name, q->getName());
  89. EXPECT_EQ(RRType::A(), q->getType());
  90. EXPECT_EQ(RRClass::IN(), q->getClass());
  91. EXPECT_EQ(1, message_parse.getRRCount(Section::QUESTION()));
  92. EXPECT_EQ(2, message_parse.getRRCount(Section::ANSWER()));
  93. EXPECT_EQ(0, message_parse.getRRCount(Section::AUTHORITY()));
  94. EXPECT_EQ(0, message_parse.getRRCount(Section::ADDITIONAL()));
  95. RRsetPtr rrset = *message_parse.beginSection(Section::ANSWER());
  96. EXPECT_EQ(test_name, rrset->getName());
  97. EXPECT_EQ(RRType::A(), rrset->getType());
  98. EXPECT_EQ(RRClass::IN(), rrset->getClass());
  99. // TTL should be 3600, even though that of the 2nd RR is 7200
  100. EXPECT_EQ(RRTTL(3600), rrset->getTTL());
  101. RdataIteratorPtr it = rrset->getRdataIterator();
  102. it->first();
  103. EXPECT_EQ("192.0.2.1", it->getCurrent().toText());
  104. it->next();
  105. EXPECT_EQ("192.0.2.2", it->getCurrent().toText());
  106. it->next();
  107. EXPECT_TRUE(it->isLast());
  108. }
  109. TEST_F(MessageTest, GetEDNS0DOBit) {
  110. // Without EDNS0, DNSSEC is considered to be unsupported.
  111. factoryFromFile(message_parse, "message_fromWire1");
  112. EXPECT_FALSE(message_parse.isDNSSECSupported());
  113. // If DO bit is on, DNSSEC is considered to be supported.
  114. message_parse.clear(Message::PARSE);
  115. factoryFromFile(message_parse, "message_fromWire2");
  116. EXPECT_TRUE(message_parse.isDNSSECSupported());
  117. // If DO bit is off, DNSSEC is considered to be unsupported.
  118. message_parse.clear(Message::PARSE);
  119. factoryFromFile(message_parse, "message_fromWire3");
  120. EXPECT_FALSE(message_parse.isDNSSECSupported());
  121. }
  122. TEST_F(MessageTest, SetEDNS0DOBit) {
  123. // By default, it's false, and we can enable/disable it.
  124. EXPECT_FALSE(message_render.isDNSSECSupported());
  125. message_render.setDNSSECSupported(true);
  126. EXPECT_TRUE(message_render.isDNSSECSupported());
  127. message_render.setDNSSECSupported(false);
  128. EXPECT_FALSE(message_render.isDNSSECSupported());
  129. // A message in the parse mode doesn't allow this flag to be set.
  130. EXPECT_THROW(message_parse.setDNSSECSupported(true),
  131. InvalidMessageOperation);
  132. // Once converted to the render mode, it works as above
  133. message_parse.makeResponse();
  134. EXPECT_FALSE(message_parse.isDNSSECSupported());
  135. message_parse.setDNSSECSupported(true);
  136. EXPECT_TRUE(message_parse.isDNSSECSupported());
  137. message_parse.setDNSSECSupported(false);
  138. EXPECT_FALSE(message_parse.isDNSSECSupported());
  139. }
  140. TEST_F(MessageTest, GetEDNS0UDPSize) {
  141. // Without EDNS0, the default max UDP size is used.
  142. factoryFromFile(message_parse, "message_fromWire1");
  143. EXPECT_EQ(Message::DEFAULT_MAX_UDPSIZE, message_parse.getUDPSize());
  144. // If the size specified in EDNS0 > default max, use it.
  145. message_parse.clear(Message::PARSE);
  146. factoryFromFile(message_parse, "message_fromWire2");
  147. EXPECT_EQ(4096, message_parse.getUDPSize());
  148. // If the size specified in EDNS0 < default max, keep using the default.
  149. message_parse.clear(Message::PARSE);
  150. factoryFromFile(message_parse, "message_fromWire8");
  151. EXPECT_EQ(Message::DEFAULT_MAX_UDPSIZE, message_parse.getUDPSize());
  152. }
  153. TEST_F(MessageTest, SetEDNS0UDPSize) {
  154. // The default size if unspecified
  155. EXPECT_EQ(Message::DEFAULT_MAX_UDPSIZE, message_render.getUDPSize());
  156. // A common buffer size with EDNS, should succeed
  157. message_render.setUDPSize(4096);
  158. EXPECT_EQ(4096, message_render.getUDPSize());
  159. // Unusual large value, but accepted
  160. message_render.setUDPSize(0xffff);
  161. EXPECT_EQ(0xffff, message_render.getUDPSize());
  162. // Too small is value is rejected
  163. EXPECT_THROW(message_render.setUDPSize(511), InvalidMessageUDPSize);
  164. // A message in the parse mode doesn't allow the set operation.
  165. EXPECT_THROW(message_parse.setUDPSize(4096), InvalidMessageOperation);
  166. // Once converted to the render mode, it works as above.
  167. message_parse.makeResponse();
  168. message_parse.setUDPSize(4096);
  169. EXPECT_EQ(4096, message_parse.getUDPSize());
  170. message_parse.setUDPSize(0xffff);
  171. EXPECT_EQ(0xffff, message_parse.getUDPSize());
  172. EXPECT_THROW(message_parse.setUDPSize(511), InvalidMessageUDPSize);
  173. }
  174. TEST_F(MessageTest, EDNS0ExtCode) {
  175. // Extended Rcode = BADVERS
  176. factoryFromFile(message_parse, "message_fromWire10");
  177. EXPECT_EQ(Rcode::BADVERS(), message_parse.getRcode());
  178. // Maximum extended Rcode
  179. message_parse.clear(Message::PARSE);
  180. factoryFromFile(message_parse, "message_fromWire11");
  181. EXPECT_EQ(0xfff, message_parse.getRcode().getCode());
  182. }
  183. TEST_F(MessageTest, BadEDNS0) {
  184. // OPT RR in the answer section
  185. EXPECT_THROW(factoryFromFile(message_parse, "message_fromWire4"),
  186. DNSMessageFORMERR);
  187. // multiple OPT RRs (in the additional section)
  188. message_parse.clear(Message::PARSE);
  189. EXPECT_THROW(factoryFromFile(message_parse, "message_fromWire5"),
  190. DNSMessageFORMERR);
  191. // OPT RR of a non root name
  192. message_parse.clear(Message::PARSE);
  193. EXPECT_THROW(factoryFromFile(message_parse, "message_fromWire6"),
  194. DNSMessageFORMERR);
  195. // Compressed owner name of OPT RR points to a root name.
  196. // Not necessarily bogus, but very unusual and mostly pathological.
  197. // We accept it, but is it okay?
  198. message_parse.clear(Message::PARSE);
  199. EXPECT_NO_THROW(factoryFromFile(message_parse, "message_fromWire7"));
  200. // Unsupported Version
  201. message_parse.clear(Message::PARSE);
  202. EXPECT_THROW(factoryFromFile(message_parse, "message_fromWire9"),
  203. DNSMessageBADVERS);
  204. }
  205. TEST_F(MessageTest, toWire) {
  206. message_render.setQid(0x1035);
  207. message_render.setOpcode(Opcode::QUERY());
  208. message_render.setRcode(Rcode::NOERROR());
  209. message_render.setHeaderFlag(MessageFlag::QR());
  210. message_render.setHeaderFlag(MessageFlag::RD());
  211. message_render.setHeaderFlag(MessageFlag::AA());
  212. message_render.addQuestion(Question(Name("test.example.com"), RRClass::IN(),
  213. RRType::A()));
  214. RRsetPtr rrset = RRsetPtr(new RRset(Name("test.example.com"), RRClass::IN(),
  215. RRType::A(), RRTTL(3600)));
  216. rrset->addRdata(in::A("192.0.2.1"));
  217. rrset->addRdata(in::A("192.0.2.2"));
  218. message_render.addRRset(Section::ANSWER(), rrset);
  219. EXPECT_EQ(1, message_render.getRRCount(Section::QUESTION()));
  220. EXPECT_EQ(2, message_render.getRRCount(Section::ANSWER()));
  221. EXPECT_EQ(0, message_render.getRRCount(Section::AUTHORITY()));
  222. EXPECT_EQ(0, message_render.getRRCount(Section::ADDITIONAL()));
  223. message_render.toWire(renderer);
  224. vector<unsigned char> data;
  225. UnitTestUtil::readWireData("message_toWire1", data);
  226. EXPECT_PRED_FORMAT4(UnitTestUtil::matchWireData, obuffer.getData(),
  227. obuffer.getLength(), &data[0], data.size());
  228. }
  229. }