ares-test-parse-mx.cc 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. #include "ares-test.h"
  2. #include "dns-proto.h"
  3. #include <sstream>
  4. #include <vector>
  5. namespace ares {
  6. namespace test {
  7. TEST_F(LibraryTest, ParseMxReplyOK) {
  8. DNSPacket pkt;
  9. pkt.set_qid(0x1234).set_response().set_aa()
  10. .add_question(new DNSQuestion("example.com", T_MX))
  11. .add_answer(new DNSMxRR("example.com", 100, 100, "mx1.example.com"))
  12. .add_answer(new DNSMxRR("example.com", 100, 200, "mx2.example.com"));
  13. std::vector<byte> data = pkt.data();
  14. struct ares_mx_reply* mx = nullptr;
  15. EXPECT_EQ(ARES_SUCCESS, ares_parse_mx_reply(data.data(), data.size(), &mx));
  16. ASSERT_NE(nullptr, mx);
  17. EXPECT_EQ("mx1.example.com", std::string(mx->host));
  18. EXPECT_EQ(100, mx->priority);
  19. struct ares_mx_reply* mx2 = mx->next;
  20. ASSERT_NE(nullptr, mx2);
  21. EXPECT_EQ("mx2.example.com", std::string(mx2->host));
  22. EXPECT_EQ(200, mx2->priority);
  23. EXPECT_EQ(nullptr, mx2->next);
  24. ares_free_data(mx);
  25. }
  26. TEST_F(LibraryTest, ParseMxReplyMalformed) {
  27. std::vector<byte> data = {
  28. 0x12, 0x34, // qid
  29. 0x84, // response + query + AA + not-TC + not-RD
  30. 0x00, // not-RA + not-Z + not-AD + not-CD + rc=NoError
  31. 0x00, 0x01, // num questions
  32. 0x00, 0x01, // num answer RRs
  33. 0x00, 0x00, // num authority RRs
  34. 0x00, 0x00, // num additional RRs
  35. // Question
  36. 0x07, 'e', 'x', 'a', 'm', 'p', 'l', 'e',
  37. 0x03, 'c', 'o', 'm',
  38. 0x00,
  39. 0x00, 0x0F, // type MX
  40. 0x00, 0x01, // class IN
  41. // Answer 1
  42. 0x07, 'e', 'x', 'a', 'm', 'p', 'l', 'e',
  43. 0x03, 'c', 'o', 'm',
  44. 0x00,
  45. 0x00, 0x0F, // RR type
  46. 0x00, 0x01, // class IN
  47. 0x01, 0x02, 0x03, 0x04, // TTL
  48. 0x00, 0x01, // rdata length -- too short
  49. 0x02,
  50. };
  51. struct ares_mx_reply* mx = nullptr;
  52. EXPECT_EQ(ARES_EBADRESP, ares_parse_mx_reply(data.data(), data.size(), &mx));
  53. ASSERT_EQ(nullptr, mx);
  54. }
  55. TEST_F(LibraryTest, ParseMxReplyErrors) {
  56. DNSPacket pkt;
  57. pkt.set_qid(0x1234).set_response().set_aa()
  58. .add_question(new DNSQuestion("example.com", T_MX))
  59. .add_answer(new DNSMxRR("example.com", 100, 100, "mx1.example.com"));
  60. std::vector<byte> data;
  61. struct ares_mx_reply* mx = nullptr;
  62. // No question.
  63. pkt.questions_.clear();
  64. data = pkt.data();
  65. EXPECT_EQ(ARES_EBADRESP, ares_parse_mx_reply(data.data(), data.size(), &mx));
  66. EXPECT_EQ(nullptr, mx);
  67. pkt.add_question(new DNSQuestion("example.com", T_MX));
  68. #ifdef DISABLED
  69. // Question != answer
  70. pkt.questions_.clear();
  71. pkt.add_question(new DNSQuestion("Axample.com", T_MX));
  72. data = pkt.data();
  73. EXPECT_EQ(ARES_EBADRESP, ares_parse_mx_reply(data.data(), data.size(), &mx));
  74. pkt.questions_.clear();
  75. pkt.add_question(new DNSQuestion("example.com", T_MX));
  76. #endif
  77. // Two questions.
  78. pkt.add_question(new DNSQuestion("example.com", T_MX));
  79. data = pkt.data();
  80. EXPECT_EQ(ARES_EBADRESP, ares_parse_mx_reply(data.data(), data.size(), &mx));
  81. EXPECT_EQ(nullptr, mx);
  82. pkt.questions_.clear();
  83. pkt.add_question(new DNSQuestion("example.com", T_MX));
  84. // Wrong sort of answer.
  85. // TODO(drysdale): check if this should be ARES_ENODATA?
  86. pkt.answers_.clear();
  87. pkt.add_answer(new DNSSrvRR("example.abc.def.com", 180, 0, 10, 8160, "example.abc.def.com"));
  88. data = pkt.data();
  89. EXPECT_EQ(ARES_SUCCESS, ares_parse_mx_reply(data.data(), data.size(), &mx));
  90. EXPECT_EQ(nullptr, mx);
  91. pkt.answers_.clear();
  92. pkt.add_answer(new DNSMxRR("example.com", 100, 100, "mx1.example.com"));
  93. // No answer.
  94. pkt.answers_.clear();
  95. data = pkt.data();
  96. EXPECT_EQ(ARES_ENODATA, ares_parse_mx_reply(data.data(), data.size(), &mx));
  97. EXPECT_EQ(nullptr, mx);
  98. pkt.add_answer(new DNSMxRR("example.com", 100, 100, "mx1.example.com"));
  99. // Truncated packets.
  100. data = pkt.data();
  101. for (size_t len = 1; len < data.size(); len++) {
  102. int rc = ares_parse_mx_reply(data.data(), len, &mx);
  103. EXPECT_EQ(nullptr, mx);
  104. EXPECT_TRUE(rc == ARES_EBADRESP || rc == ARES_EBADNAME);
  105. }
  106. }
  107. TEST_F(LibraryTest, ParseMxReplyAllocFail) {
  108. DNSPacket pkt;
  109. pkt.set_qid(0x1234).set_response().set_aa()
  110. .add_question(new DNSQuestion("example.com", T_MX))
  111. .add_answer(new DNSCnameRR("example.com", 300, "c.example.com"))
  112. .add_answer(new DNSMxRR("c.example.com", 100, 100, "mx1.example.com"));
  113. std::vector<byte> data = pkt.data();
  114. struct ares_mx_reply* mx = nullptr;
  115. for (int ii = 1; ii <= 5; ii++) {
  116. ClearFails();
  117. SetAllocFail(ii);
  118. EXPECT_EQ(ARES_ENOMEM, ares_parse_mx_reply(data.data(), data.size(), &mx)) << ii;
  119. }
  120. }
  121. } // namespace test
  122. } // namespace ares