mirror of https://github.com/c-ares/c-ares.git
Parse SOA records from ns_t_any response (#103)
Added the capability of parsing SOA record from a response buffer of ns_t_any type query, this implementation doesn't interfere with existing T_SOA query's response as that too is treated as a list of records. The function returns ARES_EBADRESP if no SOA record is found(as per RFC). The basic idea of sticking to RFC that a ns_t_any too should return an SOA record is something open for discussion but I have kept the functionality intact as it was previously i.e the function returns ARES_EBADRESP if it doesn't find a SOA record regardless of which response it is parsing i.e. T_SOA or T_ANY. Note that asking for T_ANY is generally a bad idea: - https://blog.cloudflare.com/what-happened-next-the-deprecation-of-any/ - https://tools.ietf.org/html/draft-ietf-dnsop-refuse-any Bug: #102 Fix By: Dron Rathore (@DronRathore)pull/310/head
parent
49894389fd
commit
e0517f97d9
4 changed files with 210 additions and 50 deletions
@ -0,0 +1,111 @@ |
||||
#include "ares-test.h" |
||||
#include "dns-proto.h" |
||||
|
||||
#include <sstream> |
||||
#include <vector> |
||||
|
||||
namespace ares { |
||||
namespace test { |
||||
|
||||
TEST_F(LibraryTest, ParseSoaAnyReplyOK) { |
||||
DNSPacket pkt; |
||||
pkt.set_qid(0x1234).set_response().set_aa() |
||||
.add_question(new DNSQuestion("example.com", ns_t_any))\
|
||||
.add_answer(new DNSARR("example.com", 0x01020304, {2,3,4,5})) |
||||
.add_answer(new DNSMxRR("example.com", 100, 100, "mx1.example.com")) |
||||
.add_answer(new DNSMxRR("example.com", 100, 200, "mx2.example.com")) |
||||
.add_answer(new DNSSoaRR("example.com", 100, |
||||
"soa1.example.com", "fred.example.com", |
||||
1, 2, 3, 4, 5)); |
||||
std::vector<byte> data = pkt.data(); |
||||
|
||||
struct ares_soa_reply* soa = nullptr; |
||||
EXPECT_EQ(ARES_SUCCESS, ares_parse_soa_reply(data.data(), data.size(), &soa)); |
||||
ASSERT_NE(nullptr, soa); |
||||
EXPECT_EQ("soa1.example.com", std::string(soa->nsname)); |
||||
EXPECT_EQ("fred.example.com", std::string(soa->hostmaster)); |
||||
EXPECT_EQ(1, soa->serial); |
||||
EXPECT_EQ(2, soa->refresh); |
||||
EXPECT_EQ(3, soa->retry); |
||||
EXPECT_EQ(4, soa->expire); |
||||
EXPECT_EQ(5, soa->minttl); |
||||
ares_free_data(soa); |
||||
} |
||||
|
||||
TEST_F(LibraryTest, ParseSoaAnyReplyErrors) { |
||||
DNSPacket pkt; |
||||
pkt.set_qid(0x1234).set_response().set_aa() |
||||
.add_question(new DNSQuestion("example.com", ns_t_any)) |
||||
.add_answer(new DNSSoaRR("example.com", 100, |
||||
"soa1.example.com", "fred.example.com", |
||||
1, 2, 3, 4, 5)); |
||||
std::vector<byte> data; |
||||
struct ares_soa_reply* soa = nullptr; |
||||
|
||||
// No question.
|
||||
pkt.questions_.clear(); |
||||
data = pkt.data(); |
||||
EXPECT_EQ(ARES_EBADRESP, ares_parse_soa_reply(data.data(), data.size(), &soa)); |
||||
pkt.add_question(new DNSQuestion("example.com", ns_t_any)); |
||||
|
||||
#ifdef DISABLED |
||||
// Question != answer
|
||||
pkt.questions_.clear(); |
||||
pkt.add_question(new DNSQuestion("Axample.com", ns_t_any)); |
||||
data = pkt.data(); |
||||
EXPECT_EQ(ARES_EBADRESP, ares_parse_soa_reply(data.data(), data.size(), &soa)); |
||||
pkt.questions_.clear(); |
||||
pkt.add_question(new DNSQuestion("example.com", ns_t_any)); |
||||
#endif |
||||
|
||||
// Two questions
|
||||
pkt.add_question(new DNSQuestion("example.com", ns_t_any)); |
||||
data = pkt.data(); |
||||
EXPECT_EQ(ARES_EBADRESP, ares_parse_soa_reply(data.data(), data.size(), &soa)); |
||||
pkt.questions_.clear(); |
||||
pkt.add_question(new DNSQuestion("example.com", ns_t_any)); |
||||
|
||||
// Wrong sort of answer.
|
||||
pkt.answers_.clear(); |
||||
pkt.add_answer(new DNSMxRR("example.com", 100, 100, "mx1.example.com")); |
||||
data = pkt.data(); |
||||
EXPECT_EQ(ARES_EBADRESP, ares_parse_soa_reply(data.data(), data.size(), &soa)); |
||||
pkt.answers_.clear(); |
||||
pkt.add_answer(new DNSSoaRR("example.com", 100, |
||||
"soa1.example.com", "fred.example.com", |
||||
1, 2, 3, 4, 5)); |
||||
|
||||
// No answer.
|
||||
pkt.answers_.clear(); |
||||
data = pkt.data(); |
||||
EXPECT_EQ(ARES_EBADRESP, ares_parse_soa_reply(data.data(), data.size(), &soa)); |
||||
pkt.add_answer(new DNSSoaRR("example.com", 100, |
||||
"soa1.example.com", "fred.example.com", |
||||
1, 2, 3, 4, 5)); |
||||
|
||||
// Truncated packets.
|
||||
data = pkt.data(); |
||||
for (size_t len = 1; len < data.size(); len++) { |
||||
EXPECT_EQ(ARES_EBADRESP, ares_parse_soa_reply(data.data(), len, &soa)); |
||||
} |
||||
} |
||||
|
||||
TEST_F(LibraryTest, ParseSoaAnyReplyAllocFail) { |
||||
DNSPacket pkt; |
||||
pkt.set_qid(0x1234).set_response().set_aa() |
||||
.add_question(new DNSQuestion("example.com", ns_t_any)) |
||||
.add_answer(new DNSSoaRR("example.com", 100, |
||||
"soa1.example.com", "fred.example.com", |
||||
1, 2, 3, 4, 5)); |
||||
std::vector<byte> data = pkt.data(); |
||||
struct ares_soa_reply* soa = nullptr; |
||||
|
||||
for (int ii = 1; ii <= 5; ii++) { |
||||
ClearFails(); |
||||
SetAllocFail(ii); |
||||
EXPECT_EQ(ARES_ENOMEM, ares_parse_soa_reply(data.data(), data.size(), &soa)) << ii; |
||||
} |
||||
} |
||||
|
||||
} // namespace test
|
||||
} // namespace ares
|
Loading…
Reference in new issue