message_unittest.cc 45 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060
  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. #include <fstream>
  15. #include <boost/scoped_ptr.hpp>
  16. #include <exceptions/exceptions.h>
  17. #include <util/buffer.h>
  18. #include <util/time_utilities.h>
  19. #include <util/unittests/testdata.h>
  20. #include <util/unittests/textdata.h>
  21. #include <dns/edns.h>
  22. #include <dns/exceptions.h>
  23. #include <dns/message.h>
  24. #include <dns/messagerenderer.h>
  25. #include <dns/question.h>
  26. #include <dns/opcode.h>
  27. #include <dns/rcode.h>
  28. #include <dns/rdataclass.h>
  29. #include <dns/rrclass.h>
  30. #include <dns/rrttl.h>
  31. #include <dns/rrtype.h>
  32. #include <dns/tsig.h>
  33. #include <dns/tsigkey.h>
  34. #include <gtest/gtest.h>
  35. #include <dns/tests/unittest_util.h>
  36. using isc::UnitTestUtil;
  37. using namespace std;
  38. using namespace isc;
  39. using namespace isc::dns;
  40. using namespace isc::util;
  41. using namespace isc::dns::rdata;
  42. //
  43. // Note: we need more tests, including:
  44. // parsing malformed headers
  45. // more complete tests about parsing/rendering header flags, opcode, rcode, etc.
  46. // tests for adding RRsets
  47. // tests for RRset/Question iterators
  48. // But, we'll ship with the current set of tests for now, partly because many
  49. // of the above are covered as part of other tests, and partly due to time
  50. // limitation. We also expect to revisit the fundamental design of the Message
  51. // class, at which point we'll also revise the tests including more cases.
  52. //
  53. const uint16_t Message::DEFAULT_MAX_UDPSIZE;
  54. namespace isc {
  55. namespace util {
  56. namespace detail {
  57. extern int64_t (*gettimeFunction)();
  58. }
  59. }
  60. }
  61. // XXX: this is defined as class static constants, but some compilers
  62. // seemingly cannot find the symbol when used in the EXPECT_xxx macros.
  63. const uint16_t TSIGContext::DEFAULT_FUDGE;
  64. namespace {
  65. class MessageTest : public ::testing::Test {
  66. protected:
  67. MessageTest() : test_name("test.example.com"), obuffer(0),
  68. renderer(obuffer),
  69. message_parse(Message::PARSE),
  70. message_render(Message::RENDER),
  71. bogus_section(static_cast<Message::Section>(
  72. Message::SECTION_ADDITIONAL + 1)),
  73. tsig_ctx(TSIGKey("www.example.com:"
  74. "SFuWd/q99SzF8Yzd1QbB9g=="))
  75. {
  76. rrset_a = RRsetPtr(new RRset(test_name, RRClass::IN(),
  77. RRType::A(), RRTTL(3600)));
  78. rrset_a->addRdata(in::A("192.0.2.1"));
  79. rrset_a->addRdata(in::A("192.0.2.2"));
  80. rrset_aaaa = RRsetPtr(new RRset(test_name, RRClass::IN(),
  81. RRType::AAAA(), RRTTL(3600)));
  82. rrset_aaaa->addRdata(in::AAAA("2001:db8::1234"));
  83. rrset_rrsig = RRsetPtr(new RRset(test_name, RRClass::IN(),
  84. RRType::RRSIG(), RRTTL(3600)));
  85. rrset_rrsig->addRdata(generic::RRSIG("AAAA 5 3 7200 20100322084538 "
  86. "20100220084538 1 example.com "
  87. "FAKEFAKEFAKEFAKE"));
  88. rrset_aaaa->addRRsig(rrset_rrsig);
  89. }
  90. static Question factoryFromFile(const char* datafile);
  91. const Name test_name;
  92. OutputBuffer obuffer;
  93. MessageRenderer renderer;
  94. Message message_parse;
  95. Message message_render;
  96. const Message::Section bogus_section;
  97. RRsetPtr rrset_a; // A RRset with two RDATAs
  98. RRsetPtr rrset_aaaa; // AAAA RRset with one RDATA with RRSIG
  99. RRsetPtr rrset_rrsig; // RRSIG for the AAAA RRset
  100. TSIGContext tsig_ctx;
  101. vector<unsigned char> received_data;
  102. vector<unsigned char> expected_data;
  103. void factoryFromFile(Message& message, const char* datafile,
  104. Message::ParseOptions options =
  105. Message::PARSE_DEFAULT);
  106. };
  107. void
  108. MessageTest::factoryFromFile(Message& message, const char* datafile,
  109. Message::ParseOptions options)
  110. {
  111. received_data.clear();
  112. UnitTestUtil::readWireData(datafile, received_data);
  113. InputBuffer buffer(&received_data[0], received_data.size());
  114. message.fromWire(buffer, options);
  115. }
  116. TEST_F(MessageTest, headerFlag) {
  117. // by default no flag is set
  118. EXPECT_FALSE(message_render.getHeaderFlag(Message::HEADERFLAG_QR));
  119. EXPECT_FALSE(message_render.getHeaderFlag(Message::HEADERFLAG_AA));
  120. EXPECT_FALSE(message_render.getHeaderFlag(Message::HEADERFLAG_TC));
  121. EXPECT_FALSE(message_render.getHeaderFlag(Message::HEADERFLAG_RD));
  122. EXPECT_FALSE(message_render.getHeaderFlag(Message::HEADERFLAG_RA));
  123. EXPECT_FALSE(message_render.getHeaderFlag(Message::HEADERFLAG_AD));
  124. EXPECT_FALSE(message_render.getHeaderFlag(Message::HEADERFLAG_CD));
  125. // set operation: by default it will be on
  126. message_render.setHeaderFlag(Message::HEADERFLAG_QR);
  127. EXPECT_TRUE(message_render.getHeaderFlag(Message::HEADERFLAG_QR));
  128. // it can be set to on explicitly, too
  129. message_render.setHeaderFlag(Message::HEADERFLAG_AA, true);
  130. EXPECT_TRUE(message_render.getHeaderFlag(Message::HEADERFLAG_AA));
  131. // the bit can also be cleared
  132. message_render.setHeaderFlag(Message::HEADERFLAG_AA, false);
  133. EXPECT_FALSE(message_render.getHeaderFlag(Message::HEADERFLAG_AA));
  134. // Invalid flag values
  135. EXPECT_THROW(message_render.setHeaderFlag(
  136. static_cast<Message::HeaderFlag>(0)), InvalidParameter);
  137. EXPECT_THROW(message_render.setHeaderFlag(
  138. static_cast<Message::HeaderFlag>(0x7000)),
  139. InvalidParameter);
  140. EXPECT_THROW(message_render.setHeaderFlag(
  141. static_cast<Message::HeaderFlag>(0x0800)),
  142. InvalidParameter);
  143. EXPECT_THROW(message_render.setHeaderFlag(
  144. static_cast<Message::HeaderFlag>(0x0040)),
  145. InvalidParameter);
  146. EXPECT_THROW(message_render.setHeaderFlag(
  147. static_cast<Message::HeaderFlag>(0x10000)),
  148. InvalidParameter);
  149. EXPECT_THROW(message_render.setHeaderFlag(
  150. static_cast<Message::HeaderFlag>(0x80000000)),
  151. InvalidParameter);
  152. // set operation isn't allowed in the parse mode.
  153. EXPECT_THROW(message_parse.setHeaderFlag(Message::HEADERFLAG_QR),
  154. InvalidMessageOperation);
  155. }
  156. TEST_F(MessageTest, getEDNS) {
  157. EXPECT_FALSE(message_parse.getEDNS()); // by default EDNS isn't set
  158. factoryFromFile(message_parse, "message_fromWire10.wire");
  159. EXPECT_TRUE(message_parse.getEDNS());
  160. EXPECT_EQ(0, message_parse.getEDNS()->getVersion());
  161. EXPECT_EQ(4096, message_parse.getEDNS()->getUDPSize());
  162. EXPECT_TRUE(message_parse.getEDNS()->getDNSSECAwareness());
  163. }
  164. TEST_F(MessageTest, setEDNS) {
  165. // setEDNS() isn't allowed in the parse mode
  166. EXPECT_THROW(message_parse.setEDNS(EDNSPtr(new EDNS())),
  167. InvalidMessageOperation);
  168. EDNSPtr edns = EDNSPtr(new EDNS());
  169. message_render.setEDNS(edns);
  170. EXPECT_EQ(edns, message_render.getEDNS());
  171. }
  172. TEST_F(MessageTest, fromWireWithTSIG) {
  173. // Initially there should be no TSIG
  174. EXPECT_EQ(static_cast<void*>(NULL), message_parse.getTSIGRecord());
  175. // getTSIGRecord() is only valid in the parse mode.
  176. EXPECT_THROW(message_render.getTSIGRecord(), InvalidMessageOperation);
  177. factoryFromFile(message_parse, "message_toWire2.wire");
  178. const uint8_t expected_mac[] = {
  179. 0x22, 0x70, 0x26, 0xad, 0x29, 0x7b, 0xee, 0xe7,
  180. 0x21, 0xce, 0x6c, 0x6f, 0xff, 0x1e, 0x9e, 0xf3
  181. };
  182. const TSIGRecord* tsig_rr = message_parse.getTSIGRecord();
  183. ASSERT_NE(static_cast<void*>(NULL), tsig_rr);
  184. EXPECT_EQ(Name("www.example.com"), tsig_rr->getName());
  185. EXPECT_EQ(85, tsig_rr->getLength()); // see TSIGRecordTest.getLength
  186. EXPECT_EQ(TSIGKey::HMACMD5_NAME(), tsig_rr->getRdata().getAlgorithm());
  187. EXPECT_EQ(0x4da8877a, tsig_rr->getRdata().getTimeSigned());
  188. EXPECT_EQ(TSIGContext::DEFAULT_FUDGE, tsig_rr->getRdata().getFudge());
  189. EXPECT_PRED_FORMAT4(UnitTestUtil::matchWireData,
  190. tsig_rr->getRdata().getMAC(),
  191. tsig_rr->getRdata().getMACSize(),
  192. expected_mac, sizeof(expected_mac));
  193. EXPECT_EQ(0, tsig_rr->getRdata().getError());
  194. EXPECT_EQ(0, tsig_rr->getRdata().getOtherLen());
  195. EXPECT_EQ(static_cast<void*>(NULL), tsig_rr->getRdata().getOtherData());
  196. // If we clear the message for reuse, the recorded TSIG will be cleared.
  197. message_parse.clear(Message::PARSE);
  198. EXPECT_EQ(static_cast<void*>(NULL), message_parse.getTSIGRecord());
  199. }
  200. TEST_F(MessageTest, fromWireWithTSIGCompressed) {
  201. // Mostly same as fromWireWithTSIG, but the TSIG owner name is compressed.
  202. factoryFromFile(message_parse, "message_fromWire12.wire");
  203. const TSIGRecord* tsig_rr = message_parse.getTSIGRecord();
  204. ASSERT_NE(static_cast<void*>(NULL), tsig_rr);
  205. EXPECT_EQ(Name("www.example.com"), tsig_rr->getName());
  206. // len(www.example.com) = 17, but when fully compressed, the length is
  207. // 2 bytes. So the length of the record should be 15 bytes shorter.
  208. EXPECT_EQ(70, tsig_rr->getLength());
  209. }
  210. TEST_F(MessageTest, fromWireWithBadTSIG) {
  211. // Multiple TSIG RRs
  212. EXPECT_THROW(factoryFromFile(message_parse, "message_fromWire13.wire"),
  213. DNSMessageFORMERR);
  214. message_parse.clear(Message::PARSE);
  215. // TSIG in the answer section (must be in additional)
  216. EXPECT_THROW(factoryFromFile(message_parse, "message_fromWire14.wire"),
  217. DNSMessageFORMERR);
  218. message_parse.clear(Message::PARSE);
  219. // TSIG is not the last record.
  220. EXPECT_THROW(factoryFromFile(message_parse, "message_fromWire15.wire"),
  221. DNSMessageFORMERR);
  222. message_parse.clear(Message::PARSE);
  223. // Unexpected RR Class (this will fail in constructing TSIGRecord)
  224. EXPECT_THROW(factoryFromFile(message_parse, "message_fromWire16.wire"),
  225. DNSMessageFORMERR);
  226. }
  227. TEST_F(MessageTest, getRRCount) {
  228. // by default all counters should be 0
  229. EXPECT_EQ(0, message_render.getRRCount(Message::SECTION_QUESTION));
  230. EXPECT_EQ(0, message_render.getRRCount(Message::SECTION_ANSWER));
  231. EXPECT_EQ(0, message_render.getRRCount(Message::SECTION_AUTHORITY));
  232. EXPECT_EQ(0, message_render.getRRCount(Message::SECTION_ADDITIONAL));
  233. message_render.addQuestion(Question(Name("test.example.com"),
  234. RRClass::IN(), RRType::A()));
  235. EXPECT_EQ(1, message_render.getRRCount(Message::SECTION_QUESTION));
  236. // rrset_a contains two RRs
  237. message_render.addRRset(Message::SECTION_ANSWER, rrset_a);
  238. EXPECT_EQ(2, message_render.getRRCount(Message::SECTION_ANSWER));
  239. // parse a message containing a Question and EDNS OPT RR.
  240. // OPT shouldn't be counted as normal RR, so result of getRRCount
  241. // shouldn't change.
  242. factoryFromFile(message_parse, "message_fromWire11.wire");
  243. EXPECT_EQ(1, message_render.getRRCount(Message::SECTION_QUESTION));
  244. EXPECT_EQ(0, message_render.getRRCount(Message::SECTION_ADDITIONAL));
  245. // out-of-band section ID
  246. EXPECT_THROW(message_parse.getRRCount(bogus_section), OutOfRange);
  247. }
  248. TEST_F(MessageTest, addRRset) {
  249. // default case
  250. message_render.addRRset(Message::SECTION_ANSWER, rrset_a);
  251. EXPECT_EQ(rrset_a,
  252. *message_render.beginSection(Message::SECTION_ANSWER));
  253. EXPECT_EQ(2, message_render.getRRCount(Message::SECTION_ANSWER));
  254. // signed RRset, default case
  255. message_render.clear(Message::RENDER);
  256. message_render.addRRset(Message::SECTION_ANSWER, rrset_aaaa);
  257. EXPECT_EQ(rrset_aaaa,
  258. *message_render.beginSection(Message::SECTION_ANSWER));
  259. EXPECT_EQ(1, message_render.getRRCount(Message::SECTION_ANSWER));
  260. // signed RRset, add with the RRSIG. getRRCount() should return 2
  261. message_render.clear(Message::RENDER);
  262. message_render.addRRset(Message::SECTION_ANSWER, rrset_aaaa, true);
  263. EXPECT_EQ(rrset_aaaa,
  264. *message_render.beginSection(Message::SECTION_ANSWER));
  265. EXPECT_EQ(2, message_render.getRRCount(Message::SECTION_ANSWER));
  266. // signed RRset, add explicitly without RRSIG.
  267. message_render.clear(Message::RENDER);
  268. message_render.addRRset(Message::SECTION_ANSWER, rrset_aaaa, false);
  269. EXPECT_EQ(rrset_aaaa,
  270. *message_render.beginSection(Message::SECTION_ANSWER));
  271. EXPECT_EQ(1, message_render.getRRCount(Message::SECTION_ANSWER));
  272. }
  273. TEST_F(MessageTest, badAddRRset) {
  274. // addRRset() isn't allowed in the parse mode.
  275. EXPECT_THROW(message_parse.addRRset(Message::SECTION_ANSWER,
  276. rrset_a), InvalidMessageOperation);
  277. // out-of-band section ID
  278. EXPECT_THROW(message_render.addRRset(bogus_section, rrset_a), OutOfRange);
  279. }
  280. TEST_F(MessageTest, hasRRset) {
  281. message_render.addRRset(Message::SECTION_ANSWER, rrset_a);
  282. EXPECT_TRUE(message_render.hasRRset(Message::SECTION_ANSWER, test_name,
  283. RRClass::IN(), RRType::A()));
  284. // section doesn't match
  285. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ADDITIONAL, test_name,
  286. RRClass::IN(), RRType::A()));
  287. // name doesn't match
  288. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ANSWER,
  289. Name("nomatch.example"),
  290. RRClass::IN(), RRType::A()));
  291. // RR class doesn't match
  292. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ANSWER, test_name,
  293. RRClass::CH(), RRType::A()));
  294. // RR type doesn't match
  295. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ANSWER, test_name,
  296. RRClass::IN(), RRType::AAAA()));
  297. // out-of-band section ID
  298. EXPECT_THROW(message_render.hasRRset(bogus_section, test_name,
  299. RRClass::IN(), RRType::A()),
  300. OutOfRange);
  301. // Repeat the checks having created an RRset of the appropriate type.
  302. RRsetPtr rrs1(new RRset(test_name, RRClass::IN(), RRType::A(), RRTTL(60)));
  303. EXPECT_TRUE(message_render.hasRRset(Message::SECTION_ANSWER, rrs1));
  304. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ADDITIONAL, rrs1));
  305. RRsetPtr rrs2(new RRset(Name("nomatch.example"), RRClass::IN(), RRType::A(),
  306. RRTTL(5)));
  307. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ANSWER, rrs2));
  308. RRsetPtr rrs3(new RRset(test_name, RRClass::CH(), RRType::A(), RRTTL(60)));
  309. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ANSWER, rrs3));
  310. RRsetPtr rrs4(new RRset(test_name, RRClass::IN(), RRType::AAAA(), RRTTL(5)));
  311. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ANSWER, rrs4));
  312. RRsetPtr rrs5(new RRset(test_name, RRClass::IN(), RRType::AAAA(), RRTTL(5)));
  313. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ANSWER, rrs4));
  314. EXPECT_THROW(message_render.hasRRset(bogus_section, rrs1), OutOfRange);
  315. }
  316. TEST_F(MessageTest, removeRRset) {
  317. message_render.addRRset(Message::SECTION_ANSWER, rrset_a);
  318. message_render.addRRset(Message::SECTION_ANSWER, rrset_aaaa);
  319. EXPECT_TRUE(message_render.hasRRset(Message::SECTION_ANSWER, test_name,
  320. RRClass::IN(), RRType::A()));
  321. EXPECT_TRUE(message_render.hasRRset(Message::SECTION_ANSWER, test_name,
  322. RRClass::IN(), RRType::AAAA()));
  323. EXPECT_EQ(3, message_render.getRRCount(Message::SECTION_ANSWER));
  324. // Locate the AAAA RRset and remove it; this has one RR in it.
  325. RRsetIterator i = message_render.beginSection(Message::SECTION_ANSWER);
  326. if ((*i)->getType() == RRType::A()) {
  327. ++i;
  328. }
  329. EXPECT_EQ(RRType::AAAA(), (*i)->getType());
  330. message_render.removeRRset(Message::SECTION_ANSWER, i);
  331. EXPECT_TRUE(message_render.hasRRset(Message::SECTION_ANSWER, test_name,
  332. RRClass::IN(), RRType::A()));
  333. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ANSWER, test_name,
  334. RRClass::IN(), RRType::AAAA()));
  335. EXPECT_EQ(2, message_render.getRRCount(Message::SECTION_ANSWER));
  336. }
  337. TEST_F(MessageTest, clearQuestionSection) {
  338. QuestionPtr q(new Question(Name("www.example.com"), RRClass::IN(),
  339. RRType::A()));
  340. message_render.addQuestion(q);
  341. ASSERT_EQ(1, message_render.getRRCount(Message::SECTION_QUESTION));
  342. message_render.clearSection(Message::SECTION_QUESTION);
  343. EXPECT_EQ(0, message_render.getRRCount(Message::SECTION_QUESTION));
  344. }
  345. TEST_F(MessageTest, clearAnswerSection) {
  346. // Add two RRsets, check they are present, clear the section,
  347. // check if they are gone.
  348. message_render.addRRset(Message::SECTION_ANSWER, rrset_a);
  349. message_render.addRRset(Message::SECTION_ANSWER, rrset_aaaa);
  350. ASSERT_TRUE(message_render.hasRRset(Message::SECTION_ANSWER, test_name,
  351. RRClass::IN(), RRType::A()));
  352. ASSERT_TRUE(message_render.hasRRset(Message::SECTION_ANSWER, test_name,
  353. RRClass::IN(), RRType::AAAA()));
  354. ASSERT_EQ(3, message_render.getRRCount(Message::SECTION_ANSWER));
  355. message_render.clearSection(Message::SECTION_ANSWER);
  356. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ANSWER, test_name,
  357. RRClass::IN(), RRType::A()));
  358. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ANSWER, test_name,
  359. RRClass::IN(), RRType::AAAA()));
  360. EXPECT_EQ(0, message_render.getRRCount(Message::SECTION_ANSWER));
  361. }
  362. TEST_F(MessageTest, clearAuthoritySection) {
  363. // Add two RRsets, check they are present, clear the section,
  364. // check if they are gone.
  365. message_render.addRRset(Message::SECTION_AUTHORITY, rrset_a);
  366. message_render.addRRset(Message::SECTION_AUTHORITY, rrset_aaaa);
  367. ASSERT_TRUE(message_render.hasRRset(Message::SECTION_AUTHORITY, test_name,
  368. RRClass::IN(), RRType::A()));
  369. ASSERT_TRUE(message_render.hasRRset(Message::SECTION_AUTHORITY, test_name,
  370. RRClass::IN(), RRType::AAAA()));
  371. ASSERT_EQ(3, message_render.getRRCount(Message::SECTION_AUTHORITY));
  372. message_render.clearSection(Message::SECTION_AUTHORITY);
  373. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_AUTHORITY, test_name,
  374. RRClass::IN(), RRType::A()));
  375. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_AUTHORITY, test_name,
  376. RRClass::IN(), RRType::AAAA()));
  377. EXPECT_EQ(0, message_render.getRRCount(Message::SECTION_AUTHORITY));
  378. }
  379. TEST_F(MessageTest, clearAdditionalSection) {
  380. // Add two RRsets, check they are present, clear the section,
  381. // check if they are gone.
  382. message_render.addRRset(Message::SECTION_ADDITIONAL, rrset_a);
  383. message_render.addRRset(Message::SECTION_ADDITIONAL, rrset_aaaa);
  384. ASSERT_TRUE(message_render.hasRRset(Message::SECTION_ADDITIONAL, test_name,
  385. RRClass::IN(), RRType::A()));
  386. ASSERT_TRUE(message_render.hasRRset(Message::SECTION_ADDITIONAL, test_name,
  387. RRClass::IN(), RRType::AAAA()));
  388. ASSERT_EQ(3, message_render.getRRCount(Message::SECTION_ADDITIONAL));
  389. message_render.clearSection(Message::SECTION_ADDITIONAL);
  390. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ADDITIONAL, test_name,
  391. RRClass::IN(), RRType::A()));
  392. EXPECT_FALSE(message_render.hasRRset(Message::SECTION_ADDITIONAL, test_name,
  393. RRClass::IN(), RRType::AAAA()));
  394. EXPECT_EQ(0, message_render.getRRCount(Message::SECTION_ADDITIONAL));
  395. }
  396. TEST_F(MessageTest, badBeginSection) {
  397. // valid cases are tested via other tests
  398. EXPECT_THROW(message_render.beginSection(Message::SECTION_QUESTION),
  399. InvalidMessageSection);
  400. EXPECT_THROW(message_render.beginSection(bogus_section), OutOfRange);
  401. }
  402. TEST_F(MessageTest, badEndSection) {
  403. // valid cases are tested via other tests
  404. EXPECT_THROW(message_render.endSection(Message::SECTION_QUESTION),
  405. InvalidMessageSection);
  406. EXPECT_THROW(message_render.endSection(bogus_section), OutOfRange);
  407. }
  408. TEST_F(MessageTest, appendSection) {
  409. Message target(Message::RENDER);
  410. // Section check
  411. EXPECT_THROW(target.appendSection(bogus_section, message_render),
  412. OutOfRange);
  413. // Make sure nothing is copied if there is nothing to copy
  414. target.appendSection(Message::SECTION_QUESTION, message_render);
  415. EXPECT_EQ(0, target.getRRCount(Message::SECTION_QUESTION));
  416. target.appendSection(Message::SECTION_ANSWER, message_render);
  417. EXPECT_EQ(0, target.getRRCount(Message::SECTION_ANSWER));
  418. target.appendSection(Message::SECTION_AUTHORITY, message_render);
  419. EXPECT_EQ(0, target.getRRCount(Message::SECTION_AUTHORITY));
  420. target.appendSection(Message::SECTION_ADDITIONAL, message_render);
  421. EXPECT_EQ(0, target.getRRCount(Message::SECTION_ADDITIONAL));
  422. // Now add some data, copy again, and see if it got added
  423. message_render.addQuestion(Question(Name("test.example.com"),
  424. RRClass::IN(), RRType::A()));
  425. message_render.addRRset(Message::SECTION_ANSWER, rrset_a);
  426. message_render.addRRset(Message::SECTION_AUTHORITY, rrset_a);
  427. message_render.addRRset(Message::SECTION_ADDITIONAL, rrset_a);
  428. message_render.addRRset(Message::SECTION_ADDITIONAL, rrset_aaaa);
  429. target.appendSection(Message::SECTION_QUESTION, message_render);
  430. EXPECT_EQ(1, target.getRRCount(Message::SECTION_QUESTION));
  431. target.appendSection(Message::SECTION_ANSWER, message_render);
  432. EXPECT_EQ(2, target.getRRCount(Message::SECTION_ANSWER));
  433. EXPECT_TRUE(target.hasRRset(Message::SECTION_ANSWER, test_name,
  434. RRClass::IN(), RRType::A()));
  435. target.appendSection(Message::SECTION_AUTHORITY, message_render);
  436. EXPECT_EQ(2, target.getRRCount(Message::SECTION_AUTHORITY));
  437. EXPECT_TRUE(target.hasRRset(Message::SECTION_AUTHORITY, test_name,
  438. RRClass::IN(), RRType::A()));
  439. target.appendSection(Message::SECTION_ADDITIONAL, message_render);
  440. EXPECT_EQ(3, target.getRRCount(Message::SECTION_ADDITIONAL));
  441. EXPECT_TRUE(target.hasRRset(Message::SECTION_ADDITIONAL, test_name,
  442. RRClass::IN(), RRType::A()));
  443. EXPECT_TRUE(target.hasRRset(Message::SECTION_ADDITIONAL, test_name,
  444. RRClass::IN(), RRType::AAAA()));
  445. // One more test, test to see if the section gets added, not replaced
  446. Message source2(Message::RENDER);
  447. source2.addRRset(Message::SECTION_ANSWER, rrset_aaaa);
  448. target.appendSection(Message::SECTION_ANSWER, source2);
  449. EXPECT_EQ(3, target.getRRCount(Message::SECTION_ANSWER));
  450. EXPECT_TRUE(target.hasRRset(Message::SECTION_ANSWER, test_name,
  451. RRClass::IN(), RRType::A()));
  452. EXPECT_TRUE(target.hasRRset(Message::SECTION_ANSWER, test_name,
  453. RRClass::IN(), RRType::AAAA()));
  454. }
  455. TEST_F(MessageTest, parseHeader) {
  456. received_data.clear();
  457. UnitTestUtil::readWireData("message_fromWire1", received_data);
  458. // parseHeader() isn't allowed in the render mode.
  459. InputBuffer buffer(&received_data[0], received_data.size());
  460. EXPECT_THROW(message_render.parseHeader(buffer), InvalidMessageOperation);
  461. message_parse.parseHeader(buffer);
  462. EXPECT_EQ(0x1035, message_parse.getQid());
  463. EXPECT_EQ(Opcode::QUERY(), message_parse.getOpcode());
  464. EXPECT_EQ(Rcode::NOERROR(), message_parse.getRcode());
  465. EXPECT_TRUE(message_parse.getHeaderFlag(Message::HEADERFLAG_QR));
  466. EXPECT_TRUE(message_parse.getHeaderFlag(Message::HEADERFLAG_AA));
  467. EXPECT_FALSE(message_parse.getHeaderFlag(Message::HEADERFLAG_TC));
  468. EXPECT_TRUE(message_parse.getHeaderFlag(Message::HEADERFLAG_RD));
  469. EXPECT_FALSE(message_parse.getHeaderFlag(Message::HEADERFLAG_RA));
  470. EXPECT_FALSE(message_parse.getHeaderFlag(Message::HEADERFLAG_AD));
  471. EXPECT_FALSE(message_parse.getHeaderFlag(Message::HEADERFLAG_CD));
  472. EXPECT_EQ(1, message_parse.getRRCount(Message::SECTION_QUESTION));
  473. EXPECT_EQ(2, message_parse.getRRCount(Message::SECTION_ANSWER));
  474. EXPECT_EQ(0, message_parse.getRRCount(Message::SECTION_AUTHORITY));
  475. EXPECT_EQ(0, message_parse.getRRCount(Message::SECTION_ADDITIONAL));
  476. // Only the header part should have been examined.
  477. EXPECT_EQ(12, buffer.getPosition()); // 12 = size of the header section
  478. EXPECT_TRUE(message_parse.beginQuestion() == message_parse.endQuestion());
  479. EXPECT_TRUE(message_parse.beginSection(Message::SECTION_ANSWER) ==
  480. message_parse.endSection(Message::SECTION_ANSWER));
  481. EXPECT_TRUE(message_parse.beginSection(Message::SECTION_AUTHORITY) ==
  482. message_parse.endSection(Message::SECTION_AUTHORITY));
  483. EXPECT_TRUE(message_parse.beginSection(Message::SECTION_ADDITIONAL) ==
  484. message_parse.endSection(Message::SECTION_ADDITIONAL));
  485. }
  486. TEST_F(MessageTest, fromWire) {
  487. // fromWire() isn't allowed in the render mode.
  488. EXPECT_THROW(factoryFromFile(message_render, "message_fromWire1"),
  489. InvalidMessageOperation);
  490. factoryFromFile(message_parse, "message_fromWire1");
  491. EXPECT_EQ(0x1035, message_parse.getQid());
  492. EXPECT_EQ(Opcode::QUERY(), message_parse.getOpcode());
  493. EXPECT_EQ(Rcode::NOERROR(), message_parse.getRcode());
  494. EXPECT_TRUE(message_parse.getHeaderFlag(Message::HEADERFLAG_QR));
  495. EXPECT_TRUE(message_parse.getHeaderFlag(Message::HEADERFLAG_RD));
  496. EXPECT_TRUE(message_parse.getHeaderFlag(Message::HEADERFLAG_AA));
  497. QuestionPtr q = *message_parse.beginQuestion();
  498. EXPECT_EQ(test_name, q->getName());
  499. EXPECT_EQ(RRType::A(), q->getType());
  500. EXPECT_EQ(RRClass::IN(), q->getClass());
  501. EXPECT_EQ(1, message_parse.getRRCount(Message::SECTION_QUESTION));
  502. EXPECT_EQ(2, message_parse.getRRCount(Message::SECTION_ANSWER));
  503. EXPECT_EQ(0, message_parse.getRRCount(Message::SECTION_AUTHORITY));
  504. EXPECT_EQ(0, message_parse.getRRCount(Message::SECTION_ADDITIONAL));
  505. RRsetPtr rrset = *message_parse.beginSection(Message::SECTION_ANSWER);
  506. EXPECT_EQ(test_name, rrset->getName());
  507. EXPECT_EQ(RRType::A(), rrset->getType());
  508. EXPECT_EQ(RRClass::IN(), rrset->getClass());
  509. // TTL should be 3600, even though that of the 2nd RR is 7200
  510. EXPECT_EQ(RRTTL(3600), rrset->getTTL());
  511. RdataIteratorPtr it = rrset->getRdataIterator();
  512. EXPECT_EQ("192.0.2.1", it->getCurrent().toText());
  513. it->next();
  514. EXPECT_EQ("192.0.2.2", it->getCurrent().toText());
  515. it->next();
  516. EXPECT_TRUE(it->isLast());
  517. }
  518. TEST_F(MessageTest, fromWireShortBuffer) {
  519. // We trim a valid message (ending with an SOA RR) for one byte.
  520. // fromWire() should throw an exception while parsing the trimmed RR.
  521. UnitTestUtil::readWireData("message_fromWire22.wire", received_data);
  522. InputBuffer buffer(&received_data[0], received_data.size() - 1);
  523. EXPECT_THROW(message_parse.fromWire(buffer), InvalidBufferPosition);
  524. }
  525. TEST_F(MessageTest, fromWireCombineRRs) {
  526. // This message contains 3 RRs in the answer section in the order of
  527. // A, AAAA, A types. fromWire() should combine the two A RRs into a
  528. // single RRset by default.
  529. factoryFromFile(message_parse, "message_fromWire19.wire");
  530. RRsetIterator it = message_parse.beginSection(Message::SECTION_ANSWER);
  531. RRsetIterator it_end = message_parse.endSection(Message::SECTION_ANSWER);
  532. ASSERT_TRUE(it != it_end);
  533. EXPECT_EQ(RRType::A(), (*it)->getType());
  534. EXPECT_EQ(2, (*it)->getRdataCount());
  535. ++it;
  536. ASSERT_TRUE(it != it_end);
  537. EXPECT_EQ(RRType::AAAA(), (*it)->getType());
  538. EXPECT_EQ(1, (*it)->getRdataCount());
  539. }
  540. // A helper function for a test pattern commonly used in several tests below.
  541. void
  542. preserveRRCheck(const Message& message, Message::Section section) {
  543. RRsetIterator it = message.beginSection(section);
  544. RRsetIterator it_end = message.endSection(section);
  545. ASSERT_TRUE(it != it_end);
  546. EXPECT_EQ(RRType::A(), (*it)->getType());
  547. EXPECT_EQ(1, (*it)->getRdataCount());
  548. EXPECT_EQ("192.0.2.1", (*it)->getRdataIterator()->getCurrent().toText());
  549. ++it;
  550. ASSERT_TRUE(it != it_end);
  551. EXPECT_EQ(RRType::AAAA(), (*it)->getType());
  552. EXPECT_EQ(1, (*it)->getRdataCount());
  553. EXPECT_EQ("2001:db8::1", (*it)->getRdataIterator()->getCurrent().toText());
  554. ++it;
  555. ASSERT_TRUE(it != it_end);
  556. EXPECT_EQ(RRType::A(), (*it)->getType());
  557. EXPECT_EQ(1, (*it)->getRdataCount());
  558. EXPECT_EQ("192.0.2.2", (*it)->getRdataIterator()->getCurrent().toText());
  559. }
  560. TEST_F(MessageTest, fromWirePreserveAnswer) {
  561. // Using the same data as the previous test, but specify the PRESERVE_ORDER
  562. // option. The received order of RRs should be preserved, and each RR
  563. // should be stored in a single RRset.
  564. factoryFromFile(message_parse, "message_fromWire19.wire",
  565. Message::PRESERVE_ORDER);
  566. {
  567. SCOPED_TRACE("preserve answer RRs");
  568. preserveRRCheck(message_parse, Message::SECTION_ANSWER);
  569. }
  570. }
  571. TEST_F(MessageTest, fromWirePreserveAuthority) {
  572. // Same for the previous test, but for the authority section.
  573. factoryFromFile(message_parse, "message_fromWire20.wire",
  574. Message::PRESERVE_ORDER);
  575. {
  576. SCOPED_TRACE("preserve authority RRs");
  577. preserveRRCheck(message_parse, Message::SECTION_AUTHORITY);
  578. }
  579. }
  580. TEST_F(MessageTest, fromWirePreserveAdditional) {
  581. // Same for the previous test, but for the additional section.
  582. factoryFromFile(message_parse, "message_fromWire21.wire",
  583. Message::PRESERVE_ORDER);
  584. {
  585. SCOPED_TRACE("preserve additional RRs");
  586. preserveRRCheck(message_parse, Message::SECTION_ADDITIONAL);
  587. }
  588. }
  589. TEST_F(MessageTest, EDNS0ExtRcode) {
  590. // Extended Rcode = BADVERS
  591. factoryFromFile(message_parse, "message_fromWire10.wire");
  592. EXPECT_EQ(Rcode::BADVERS(), message_parse.getRcode());
  593. // Maximum extended Rcode
  594. message_parse.clear(Message::PARSE);
  595. factoryFromFile(message_parse, "message_fromWire11.wire");
  596. EXPECT_EQ(0xfff, message_parse.getRcode().getCode());
  597. }
  598. TEST_F(MessageTest, BadEDNS0) {
  599. // OPT RR in the answer section
  600. EXPECT_THROW(factoryFromFile(message_parse, "message_fromWire4"),
  601. DNSMessageFORMERR);
  602. // multiple OPT RRs (in the additional section)
  603. message_parse.clear(Message::PARSE);
  604. EXPECT_THROW(factoryFromFile(message_parse, "message_fromWire5"),
  605. DNSMessageFORMERR);
  606. }
  607. TEST_F(MessageTest, toWire) {
  608. message_render.setQid(0x1035);
  609. message_render.setOpcode(Opcode::QUERY());
  610. message_render.setRcode(Rcode::NOERROR());
  611. message_render.setHeaderFlag(Message::HEADERFLAG_QR, true);
  612. message_render.setHeaderFlag(Message::HEADERFLAG_RD, true);
  613. message_render.setHeaderFlag(Message::HEADERFLAG_AA, true);
  614. message_render.addQuestion(Question(Name("test.example.com"), RRClass::IN(),
  615. RRType::A()));
  616. message_render.addRRset(Message::SECTION_ANSWER, rrset_a);
  617. EXPECT_EQ(1, message_render.getRRCount(Message::SECTION_QUESTION));
  618. EXPECT_EQ(2, message_render.getRRCount(Message::SECTION_ANSWER));
  619. EXPECT_EQ(0, message_render.getRRCount(Message::SECTION_AUTHORITY));
  620. EXPECT_EQ(0, message_render.getRRCount(Message::SECTION_ADDITIONAL));
  621. message_render.toWire(renderer);
  622. vector<unsigned char> data;
  623. UnitTestUtil::readWireData("message_toWire1", data);
  624. EXPECT_PRED_FORMAT4(UnitTestUtil::matchWireData, obuffer.getData(),
  625. obuffer.getLength(), &data[0], data.size());
  626. }
  627. TEST_F(MessageTest, toWireInParseMode) {
  628. // toWire() isn't allowed in the parse mode.
  629. EXPECT_THROW(message_parse.toWire(renderer), InvalidMessageOperation);
  630. }
  631. // See dnssectime_unittest.cc
  632. template <int64_t NOW>
  633. int64_t
  634. testGetTime() {
  635. return (NOW);
  636. }
  637. // bit-wise constant flags to configure DNS header flags for test
  638. // messages.
  639. const unsigned int QR_FLAG = 0x1;
  640. const unsigned int AA_FLAG = 0x2;
  641. const unsigned int RD_FLAG = 0x4;
  642. void
  643. commonTSIGToWireCheck(Message& message, MessageRenderer& renderer,
  644. TSIGContext& tsig_ctx, const char* const expected_file,
  645. unsigned int message_flags = RD_FLAG,
  646. RRType qtype = RRType::A(),
  647. const vector<const char*>* answer_data = NULL)
  648. {
  649. message.setOpcode(Opcode::QUERY());
  650. message.setRcode(Rcode::NOERROR());
  651. if ((message_flags & QR_FLAG) != 0) {
  652. message.setHeaderFlag(Message::HEADERFLAG_QR);
  653. }
  654. if ((message_flags & AA_FLAG) != 0) {
  655. message.setHeaderFlag(Message::HEADERFLAG_AA);
  656. }
  657. if ((message_flags & RD_FLAG) != 0) {
  658. message.setHeaderFlag(Message::HEADERFLAG_RD);
  659. }
  660. message.addQuestion(Question(Name("www.example.com"), RRClass::IN(),
  661. qtype));
  662. if (answer_data != NULL) {
  663. RRsetPtr ans_rrset(new RRset(Name("www.example.com"), RRClass::IN(),
  664. qtype, RRTTL(86400)));
  665. for (vector<const char*>::const_iterator it = answer_data->begin();
  666. it != answer_data->end();
  667. ++it) {
  668. ans_rrset->addRdata(createRdata(qtype, RRClass::IN(), *it));
  669. }
  670. message.addRRset(Message::SECTION_ANSWER, ans_rrset);
  671. }
  672. message.toWire(renderer, tsig_ctx);
  673. vector<unsigned char> expected_data;
  674. UnitTestUtil::readWireData(expected_file, expected_data);
  675. EXPECT_PRED_FORMAT4(UnitTestUtil::matchWireData, renderer.getData(),
  676. renderer.getLength(),
  677. &expected_data[0], expected_data.size());
  678. }
  679. TEST_F(MessageTest, toWireWithTSIG) {
  680. // Rendering a message with TSIG. Various special cases specific to
  681. // TSIG are tested in the tsig tests. We only check the message contains
  682. // a TSIG at the end and the ARCOUNT of the header is updated.
  683. isc::util::detail::gettimeFunction = testGetTime<0x4da8877a>;
  684. message_render.setQid(0x2d65);
  685. {
  686. SCOPED_TRACE("Message sign with TSIG");
  687. commonTSIGToWireCheck(message_render, renderer, tsig_ctx,
  688. "message_toWire2.wire");
  689. }
  690. }
  691. TEST_F(MessageTest, toWireWithEDNSAndTSIG) {
  692. // Similar to the previous test, but with an EDNS before TSIG.
  693. // The wire data check will confirm the ordering.
  694. isc::util::detail::gettimeFunction = testGetTime<0x4db60d1f>;
  695. message_render.setQid(0x6cd);
  696. EDNSPtr edns(new EDNS());
  697. edns->setUDPSize(4096);
  698. message_render.setEDNS(edns);
  699. {
  700. SCOPED_TRACE("Message sign with TSIG and EDNS");
  701. commonTSIGToWireCheck(message_render, renderer, tsig_ctx,
  702. "message_toWire3.wire");
  703. }
  704. }
  705. // Some of the following tests involve truncation. We use the query name
  706. // "www.example.com" and some TXT question/answers. The length of the
  707. // header and question will be 33 bytes. If we also try to include a
  708. // TSIG of the same key name (not compressed) with HMAC-MD5, the TSIG RR
  709. // will be 85 bytes.
  710. // A long TXT RDATA. With a fully compressed owner name, the corresponding
  711. // RR will be 268 bytes.
  712. const char* const long_txt1 = "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcde";
  713. // With a fully compressed owner name, the corresponding RR will be 212 bytes.
  714. // It should result in truncation even without TSIG (33 + 268 + 212 = 513)
  715. const char* const long_txt2 = "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456";
  716. // With a fully compressed owner name, the corresponding RR will be 127 bytes.
  717. // So, it can fit in the standard 512 bytes with txt1 and without TSIG, but
  718. // adding a TSIG would result in truncation (33 + 268 + 127 + 85 = 513)
  719. const char* const long_txt3 = "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef01";
  720. // This is 1 byte shorter than txt3, which will result in a possible longest
  721. // message containing answer RRs and TSIG.
  722. const char* const long_txt4 = "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef0";
  723. // Example output generated by
  724. // "dig -y www.example.com:SFuWd/q99SzF8Yzd1QbB9g== www.example.com txt
  725. // QID: 0x22c2
  726. // Time Signed: 0x00004e179212
  727. TEST_F(MessageTest, toWireTSIGTruncation) {
  728. isc::util::detail::gettimeFunction = testGetTime<0x4e179212>;
  729. // Verify a validly signed query so that we can use the TSIG context
  730. factoryFromFile(message_parse, "message_fromWire17.wire");
  731. EXPECT_EQ(TSIGError::NOERROR(),
  732. tsig_ctx.verify(message_parse.getTSIGRecord(),
  733. &received_data[0], received_data.size()));
  734. message_render.setQid(0x22c2);
  735. vector<const char*> answer_data;
  736. answer_data.push_back(long_txt1);
  737. answer_data.push_back(long_txt2);
  738. {
  739. SCOPED_TRACE("Message sign with TSIG and TC bit on");
  740. commonTSIGToWireCheck(message_render, renderer, tsig_ctx,
  741. "message_toWire4.wire",
  742. QR_FLAG|AA_FLAG|RD_FLAG,
  743. RRType::TXT(), &answer_data);
  744. }
  745. }
  746. TEST_F(MessageTest, toWireTSIGTruncation2) {
  747. // Similar to the previous test, but without TSIG it wouldn't cause
  748. // truncation.
  749. isc::util::detail::gettimeFunction = testGetTime<0x4e179212>;
  750. factoryFromFile(message_parse, "message_fromWire17.wire");
  751. EXPECT_EQ(TSIGError::NOERROR(),
  752. tsig_ctx.verify(message_parse.getTSIGRecord(),
  753. &received_data[0], received_data.size()));
  754. message_render.setQid(0x22c2);
  755. vector<const char*> answer_data;
  756. answer_data.push_back(long_txt1);
  757. answer_data.push_back(long_txt3);
  758. {
  759. SCOPED_TRACE("Message sign with TSIG and TC bit on (2)");
  760. commonTSIGToWireCheck(message_render, renderer, tsig_ctx,
  761. "message_toWire4.wire",
  762. QR_FLAG|AA_FLAG|RD_FLAG,
  763. RRType::TXT(), &answer_data);
  764. }
  765. }
  766. TEST_F(MessageTest, toWireTSIGTruncation3) {
  767. // Similar to previous ones, but truncation occurs due to too many
  768. // Questions (very unusual, but not necessarily illegal).
  769. // We are going to create a message starting with a standard
  770. // header (12 bytes) and multiple questions in the Question
  771. // section of the same owner name (changing the RRType, just so
  772. // that it would be the form that would be accepted by the BIND 9
  773. // parser). The first Question is 21 bytes in length, and the subsequent
  774. // ones are 6 bytes. We'll also use a TSIG whose size is 85 bytes.
  775. // Up to 66 questions can fit in the standard 512-byte buffer
  776. // (12 + 21 + 6 * 65 + 85 = 508). If we try to add one more it would
  777. // result in truncation.
  778. message_render.setOpcode(Opcode::QUERY());
  779. message_render.setRcode(Rcode::NOERROR());
  780. for (int i = 1; i <= 67; ++i) {
  781. message_render.addQuestion(Question(Name("www.example.com"),
  782. RRClass::IN(), RRType(i)));
  783. }
  784. message_render.toWire(renderer, tsig_ctx);
  785. // Check the rendered data by parsing it. We only check it has the
  786. // TC bit on, has the correct number of questions, and has a TSIG RR.
  787. // Checking the signature wouldn't be necessary for this rare case
  788. // scenario.
  789. InputBuffer buffer(renderer.getData(), renderer.getLength());
  790. message_parse.fromWire(buffer);
  791. EXPECT_TRUE(message_parse.getHeaderFlag(Message::HEADERFLAG_TC));
  792. // Note that the number of questions are 66, not 67 as we tried to add.
  793. EXPECT_EQ(66, message_parse.getRRCount(Message::SECTION_QUESTION));
  794. EXPECT_TRUE(message_parse.getTSIGRecord() != NULL);
  795. }
  796. TEST_F(MessageTest, toWireTSIGNoTruncation) {
  797. // A boundary case that shouldn't cause truncation: the resulting
  798. // response message with a TSIG will be 512 bytes long.
  799. isc::util::detail::gettimeFunction = testGetTime<0x4e17b38d>;
  800. factoryFromFile(message_parse, "message_fromWire18.wire");
  801. EXPECT_EQ(TSIGError::NOERROR(),
  802. tsig_ctx.verify(message_parse.getTSIGRecord(),
  803. &received_data[0], received_data.size()));
  804. message_render.setQid(0xd6e2);
  805. vector<const char*> answer_data;
  806. answer_data.push_back(long_txt1);
  807. answer_data.push_back(long_txt4);
  808. {
  809. SCOPED_TRACE("Message sign with TSIG, no truncation");
  810. commonTSIGToWireCheck(message_render, renderer, tsig_ctx,
  811. "message_toWire5.wire",
  812. QR_FLAG|AA_FLAG|RD_FLAG,
  813. RRType::TXT(), &answer_data);
  814. }
  815. }
  816. // This is a buggy renderer for testing. It behaves like the straightforward
  817. // MessageRenderer, but once it has some data, its setLengthLimit() ignores
  818. // the given parameter and resets the limit to the current length, making
  819. // subsequent insertion result in truncation, which would make TSIG RR
  820. // rendering fail unexpectedly in the test that follows.
  821. class BadRenderer : public MessageRenderer {
  822. public:
  823. BadRenderer(isc::util::OutputBuffer& buffer) :
  824. MessageRenderer(buffer)
  825. {}
  826. virtual void setLengthLimit(size_t len) {
  827. if (getLength() > 0) {
  828. MessageRenderer::setLengthLimit(getLength());
  829. } else {
  830. MessageRenderer::setLengthLimit(len);
  831. }
  832. }
  833. };
  834. TEST_F(MessageTest, toWireTSIGLengthErrors) {
  835. // specify an unusual short limit that wouldn't be able to hold
  836. // the TSIG.
  837. renderer.setLengthLimit(tsig_ctx.getTSIGLength() - 1);
  838. // Use commonTSIGToWireCheck() only to call toWire() with otherwise valid
  839. // conditions. The checks inside it don't matter because we expect an
  840. // exception before any of the checks.
  841. EXPECT_THROW(commonTSIGToWireCheck(message_render, renderer, tsig_ctx,
  842. "message_toWire2.wire"),
  843. InvalidParameter);
  844. // This one is large enough for TSIG, but the remaining limit isn't
  845. // even enough for the Header section.
  846. renderer.clear();
  847. message_render.clear(Message::RENDER);
  848. renderer.setLengthLimit(tsig_ctx.getTSIGLength() + 1);
  849. EXPECT_THROW(commonTSIGToWireCheck(message_render, renderer, tsig_ctx,
  850. "message_toWire2.wire"),
  851. InvalidParameter);
  852. // Trying to render a message with TSIG using a buggy renderer.
  853. obuffer.clear();
  854. BadRenderer bad_renderer(obuffer);
  855. bad_renderer.setLengthLimit(512);
  856. message_render.clear(Message::RENDER);
  857. EXPECT_THROW(commonTSIGToWireCheck(message_render, bad_renderer, tsig_ctx,
  858. "message_toWire2.wire"),
  859. Unexpected);
  860. }
  861. TEST_F(MessageTest, toWireWithoutOpcode) {
  862. message_render.setRcode(Rcode::NOERROR());
  863. EXPECT_THROW(message_render.toWire(renderer), InvalidMessageOperation);
  864. }
  865. TEST_F(MessageTest, toWireWithoutRcode) {
  866. message_render.setOpcode(Opcode::QUERY());
  867. EXPECT_THROW(message_render.toWire(renderer), InvalidMessageOperation);
  868. }
  869. TEST_F(MessageTest, toText) {
  870. // Check toText() output for a typical DNS response with records in
  871. // all sections
  872. factoryFromFile(message_parse, "message_toText1.wire");
  873. {
  874. SCOPED_TRACE("Message toText test (basic case)");
  875. ifstream ifs;
  876. unittests::openTestData("message_toText1.txt", ifs);
  877. unittests::matchTextData(ifs, message_parse.toText());
  878. }
  879. // Another example with EDNS. The expected data was slightly modified
  880. // from the dig output (other than replacing tabs with a space): adding
  881. // a newline after the "OPT PSEUDOSECTION". This is an intentional change
  882. // in our version for better readability.
  883. message_parse.clear(Message::PARSE);
  884. factoryFromFile(message_parse, "message_toText2.wire");
  885. {
  886. SCOPED_TRACE("Message toText test with EDNS");
  887. ifstream ifs;
  888. unittests::openTestData("message_toText2.txt", ifs);
  889. unittests::matchTextData(ifs, message_parse.toText());
  890. }
  891. // Another example with TSIG. The expected data was slightly modified
  892. // from the dig output (other than replacing tabs with a space): removing
  893. // a redundant white space at the end of TSIG RDATA. We'd rather consider
  894. // it a dig's defect than a feature.
  895. message_parse.clear(Message::PARSE);
  896. factoryFromFile(message_parse, "message_toText3.wire");
  897. {
  898. SCOPED_TRACE("Message toText test with TSIG");
  899. ifstream ifs;
  900. unittests::openTestData("message_toText3.txt", ifs);
  901. unittests::matchTextData(ifs, message_parse.toText());
  902. }
  903. }
  904. TEST_F(MessageTest, toTextWithoutOpcode) {
  905. message_render.setRcode(Rcode::NOERROR());
  906. EXPECT_THROW(message_render.toText(), InvalidMessageOperation);
  907. }
  908. TEST_F(MessageTest, toTextWithoutRcode) {
  909. message_render.setOpcode(Opcode::QUERY());
  910. EXPECT_THROW(message_render.toText(), InvalidMessageOperation);
  911. }
  912. }