mirror of https://github.com/c-ares/c-ares.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
385 lines
14 KiB
385 lines
14 KiB
/* MIT License |
|
* |
|
* Copyright (c) The c-ares project and its contributors |
|
* |
|
* Permission is hereby granted, free of charge, to any person obtaining a copy |
|
* of this software and associated documentation files (the "Software"), to deal |
|
* in the Software without restriction, including without limitation the rights |
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
|
* copies of the Software, and to permit persons to whom the Software is |
|
* furnished to do so, subject to the following conditions: |
|
* |
|
* The above copyright notice and this permission notice (including the next |
|
* paragraph) shall be included in all copies or substantial portions of the |
|
* Software. |
|
* |
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |
|
* SOFTWARE. |
|
* |
|
* SPDX-License-Identifier: MIT |
|
*/ |
|
#include "ares-test.h" |
|
#include "dns-proto.h" |
|
|
|
#include <sstream> |
|
#include <vector> |
|
|
|
namespace ares { |
|
namespace test { |
|
|
|
TEST_F(LibraryTest, ParseAReplyOK) { |
|
DNSPacket pkt; |
|
pkt.set_qid(0x1234).set_response().set_aa() |
|
.add_question(new DNSQuestion("example.com", T_A)) |
|
.add_answer(new DNSARR("example.com", 0x01020304, {2,3,4,5})) |
|
.add_answer(new DNSAaaaRR("example.com", 0x01020304, {0,0,0,0,0,0,0,0,0,0,0,0,2,3,4,5})); |
|
std::vector<byte> data = { |
|
0x12, 0x34, // qid |
|
0x84, // response + query + AA + not-TC + not-RD |
|
0x00, // not-RA + not-Z + not-AD + not-CD + rc=NoError |
|
0x00, 0x01, // num questions |
|
0x00, 0x02, // num answer RRs |
|
0x00, 0x00, // num authority RRs |
|
0x00, 0x00, // num additional RRs |
|
// Question |
|
0x07, 'e', 'x', 'a', 'm', 'p', 'l', 'e', |
|
0x03, 'c', 'o', 'm', |
|
0x00, |
|
0x00, 0x01, // type A |
|
0x00, 0x01, // class IN |
|
// Answer 1 |
|
0x07, 'e', 'x', 'a', 'm', 'p', 'l', 'e', |
|
0x03, 'c', 'o', 'm', |
|
0x00, |
|
0x00, 0x01, // RR type |
|
0x00, 0x01, // class IN |
|
0x01, 0x02, 0x03, 0x04, // TTL |
|
0x00, 0x04, // rdata length |
|
0x02, 0x03, 0x04, 0x05, |
|
// Answer 2 |
|
0x07, 'e', 'x', 'a', 'm', 'p', 'l', 'e', |
|
0x03, 'c', 'o', 'm', |
|
0x00, |
|
0x00, 0x1c, // RR type |
|
0x00, 0x01, // class IN |
|
0x01, 0x02, 0x03, 0x04, // TTL |
|
0x00, 0x10, // rdata length |
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x03, 0x04, 0x05, |
|
}; |
|
EXPECT_EQ(data, pkt.data()); |
|
struct hostent *host = nullptr; |
|
struct ares_addrttl info[5]; |
|
int count = 5; |
|
EXPECT_EQ(ARES_SUCCESS, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
EXPECT_EQ(1, count); |
|
EXPECT_EQ(0x01020304, info[0].ttl); |
|
unsigned long expected_addr = htonl(0x02030405); |
|
EXPECT_EQ(expected_addr, info[0].ipaddr.s_addr); |
|
EXPECT_EQ("2.3.4.5", AddressToString(&(info[0].ipaddr), 4)); |
|
ASSERT_NE(nullptr, host); |
|
std::stringstream ss; |
|
ss << HostEnt(host); |
|
EXPECT_EQ("{'example.com' aliases=[] addrs=[2.3.4.5]}", ss.str()); |
|
ares_free_hostent(host); |
|
|
|
// Repeat without providing a hostent |
|
EXPECT_EQ(ARES_SUCCESS, ares_parse_a_reply(data.data(), (int)data.size(), |
|
nullptr, info, &count)); |
|
EXPECT_EQ(1, count); |
|
EXPECT_EQ(0x01020304, info[0].ttl); |
|
EXPECT_EQ(expected_addr, info[0].ipaddr.s_addr); |
|
EXPECT_EQ("2.3.4.5", AddressToString(&(info[0].ipaddr), 4)); |
|
} |
|
|
|
TEST_F(LibraryTest, ParseMalformedAReply) { |
|
std::vector<byte> data = { |
|
0x12, 0x34, // [0:2) qid |
|
0x84, // [2] response + query + AA + not-TC + not-RD |
|
0x00, // [3] not-RA + not-Z + not-AD + not-CD + rc=NoError |
|
0x00, 0x01, // [4:6) num questions |
|
0x00, 0x02, // [6:8) num answer RRs |
|
0x00, 0x00, // [8:10) num authority RRs |
|
0x00, 0x00, // [10:12) num additional RRs |
|
// Question |
|
0x07, 'e', 'x', 'a', 'm', 'p', 'l', 'e', // [12:20) |
|
0x03, 'c', 'o', 'm', // [20,24) |
|
0x00, // [24] |
|
0x00, 0x01, // [25:26) type A |
|
0x00, 0x01, // [27:29) class IN |
|
// Answer 1 |
|
0x07, 'e', 'x', 'a', 'm', 'p', 'l', 'e', // [29:37) |
|
0x03, 'c', 'o', 'm', // [37:41) |
|
0x00, // [41] |
|
0x00, 0x01, // [42:44) RR type |
|
0x00, 0x01, // [44:46) class IN |
|
0x01, 0x02, 0x03, 0x04, // [46:50) TTL |
|
0x00, 0x04, // [50:52) rdata length |
|
0x02, 0x03, 0x04, 0x05, // [52,56) |
|
}; |
|
struct hostent *host = nullptr; |
|
struct ares_addrttl info[2]; |
|
int count = 2; |
|
|
|
// Invalid RR-len. |
|
std::vector<byte> invalid_rrlen(data); |
|
invalid_rrlen[51] = 180; |
|
EXPECT_EQ(ARES_EBADRESP, ares_parse_a_reply(invalid_rrlen.data(), (int)invalid_rrlen.size(), |
|
&host, info, &count)); |
|
|
|
// Truncate mid-question. |
|
EXPECT_EQ(ARES_EBADRESP, ares_parse_a_reply(data.data(), 26, |
|
&host, info, &count)); |
|
|
|
// Truncate mid-answer. |
|
EXPECT_EQ(ARES_EBADRESP, ares_parse_a_reply(data.data(), 42, |
|
&host, info, &count)); |
|
|
|
// Negative length |
|
EXPECT_EQ(ARES_EBADRESP, ares_parse_a_reply(data.data(), -1, |
|
&host, info, &count)); |
|
} |
|
|
|
TEST_F(LibraryTest, ParseAReplyNoData) { |
|
DNSPacket pkt; |
|
pkt.set_qid(0x1234).set_response().set_aa() |
|
.add_question(new DNSQuestion("example.com", T_A)); |
|
std::vector<byte> data = pkt.data(); |
|
struct hostent *host = nullptr; |
|
struct ares_addrttl info[2]; |
|
int count = 2; |
|
EXPECT_EQ(ARES_ENODATA, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
EXPECT_EQ(0, count); |
|
EXPECT_EQ(nullptr, host); |
|
|
|
// Again but with a CNAME. |
|
pkt.add_answer(new DNSCnameRR("example.com", 200, "c.example.com")); |
|
data = pkt.data(); |
|
// Expect success as per https://github.com/c-ares/c-ares/commit/2c63440127feed70ccefb148b8f938a2df6c15f8 |
|
EXPECT_EQ(ARES_SUCCESS, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
EXPECT_EQ(0, count); |
|
EXPECT_NE(nullptr, host); |
|
std::stringstream ss; |
|
ss << HostEnt(host); |
|
EXPECT_EQ("{'c.example.com' aliases=[example.com] addrs=[]}", ss.str()); |
|
ares_free_hostent(host); |
|
} |
|
|
|
TEST_F(LibraryTest, ParseAReplyVariantA) { |
|
DNSPacket pkt; |
|
pkt.set_qid(6366).set_rd().set_ra() |
|
.add_question(new DNSQuestion("mit.edu", T_A)) |
|
.add_answer(new DNSARR("mit.edu", 52, {18,7,22,69})) |
|
.add_auth(new DNSNsRR("mit.edu", 292, "W20NS.mit.edu")) |
|
.add_auth(new DNSNsRR("mit.edu", 292, "BITSY.mit.edu")) |
|
.add_auth(new DNSNsRR("mit.edu", 292, "STRAWB.mit.edu")) |
|
.add_additional(new DNSARR("STRAWB.mit.edu", 292, {18,71,0,151})); |
|
struct hostent *host = nullptr; |
|
struct ares_addrttl info[2]; |
|
int count = 2; |
|
std::vector<byte> data = pkt.data(); |
|
EXPECT_EQ(ARES_SUCCESS, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
EXPECT_EQ(1, count); |
|
EXPECT_EQ("18.7.22.69", AddressToString(&(info[0].ipaddr), 4)); |
|
EXPECT_EQ(52, info[0].ttl); |
|
ares_free_hostent(host); |
|
} |
|
|
|
TEST_F(LibraryTest, ParseAReplyJustCname) { |
|
DNSPacket pkt; |
|
pkt.set_qid(6366).set_rd().set_ra() |
|
.add_question(new DNSQuestion("mit.edu", T_A)) |
|
.add_answer(new DNSCnameRR("mit.edu", 52, "other.mit.edu")); |
|
struct hostent *host = nullptr; |
|
struct ares_addrttl info[2]; |
|
int count = 2; |
|
std::vector<byte> data = pkt.data(); |
|
EXPECT_EQ(ARES_SUCCESS, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
EXPECT_EQ(0, count); |
|
ASSERT_NE(nullptr, host); |
|
std::stringstream ss; |
|
ss << HostEnt(host); |
|
EXPECT_EQ("{'other.mit.edu' aliases=[mit.edu] addrs=[]}", ss.str()); |
|
ares_free_hostent(host); |
|
} |
|
|
|
TEST_F(LibraryTest, ParseAReplyVariantCname) { |
|
DNSPacket pkt; |
|
pkt.set_qid(6366).set_rd().set_ra() |
|
.add_question(new DNSQuestion("query.example.com", T_A)) |
|
.add_answer(new DNSCnameRR("query.example.com", 200, "redirect.query.example.com")) |
|
.add_answer(new DNSARR("redirect.query.example.com", 300, {129,97,123,22})) |
|
.add_auth(new DNSNsRR("example.com", 218, "aa.ns1.example.com")) |
|
.add_auth(new DNSNsRR("example.com", 218, "ns2.example.com")) |
|
.add_auth(new DNSNsRR("example.com", 218, "ns3.example.com")) |
|
.add_auth(new DNSNsRR("example.com", 218, "ns4.example.com")) |
|
.add_additional(new DNSARR("aa.ns1.example.com", 218, {129,97,1,1})) |
|
.add_additional(new DNSARR("ns2.example.com", 218, {129,97,1,2})) |
|
.add_additional(new DNSARR("ns3.example.com", 218, {129,97,1,3})) |
|
.add_additional(new DNSARR("ns4.example.com", 218, {129,97,1,4})); |
|
struct hostent *host = nullptr; |
|
struct ares_addrttl info[2]; |
|
int count = 2; |
|
std::vector<byte> data = pkt.data(); |
|
EXPECT_EQ(ARES_SUCCESS, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
EXPECT_EQ(1, count); |
|
EXPECT_EQ("129.97.123.22", AddressToString(&(info[0].ipaddr), 4)); |
|
// TTL is reduced to match CNAME's. |
|
EXPECT_EQ(200, info[0].ttl); |
|
ares_free_hostent(host); |
|
|
|
// Repeat parsing without places to put the results. |
|
count = 0; |
|
EXPECT_EQ(ARES_SUCCESS, ares_parse_a_reply(data.data(), (int)data.size(), |
|
nullptr, info, &count)); |
|
} |
|
|
|
TEST_F(LibraryTest, ParseAReplyVariantCnameChain) { |
|
DNSPacket pkt; |
|
pkt.set_qid(6366).set_rd().set_ra() |
|
.add_question(new DNSQuestion("c1.localhost", T_A)) |
|
.add_answer(new DNSCnameRR("c1.localhost", 604800, "c2.localhost")) |
|
.add_answer(new DNSCnameRR("c2.localhost", 604800, "c3.localhost")) |
|
.add_answer(new DNSCnameRR("c3.localhost", 604800, "c4.localhost")) |
|
.add_answer(new DNSARR("c4.localhost", 604800, {8,8,8,8})) |
|
.add_auth(new DNSNsRR("localhost", 604800, "localhost")) |
|
.add_additional(new DNSARR("localhost", 604800, {127,0,0,1})) |
|
.add_additional(new DNSAaaaRR("localhost", 604800, |
|
{0x7F, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, |
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01})); |
|
struct hostent *host = nullptr; |
|
struct ares_addrttl info[2]; |
|
int count = 2; |
|
std::vector<byte> data = pkt.data(); |
|
EXPECT_EQ(ARES_SUCCESS, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
EXPECT_EQ(1, count); |
|
EXPECT_EQ("8.8.8.8", AddressToString(&(info[0].ipaddr), 4)); |
|
EXPECT_EQ(604800, info[0].ttl); |
|
ares_free_hostent(host); |
|
} |
|
|
|
TEST_F(LibraryTest, ParseAReplyErrors) { |
|
DNSPacket pkt; |
|
pkt.set_qid(0x1234).set_response().set_aa() |
|
.add_question(new DNSQuestion("example.com", T_A)) |
|
.add_answer(new DNSARR("example.com", 100, {0x02, 0x03, 0x04, 0x05})); |
|
std::vector<byte> data; |
|
|
|
struct hostent *host = nullptr; |
|
struct ares_addrttl info[2]; |
|
int count = 2; |
|
|
|
// No question. |
|
pkt.questions_.clear(); |
|
data = pkt.data(); |
|
EXPECT_EQ(ARES_EBADRESP, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
EXPECT_EQ(nullptr, host); |
|
pkt.add_question(new DNSQuestion("example.com", T_A)); |
|
|
|
// Question != answer, this is ok as of Issue #683 |
|
pkt.questions_.clear(); |
|
pkt.add_question(new DNSQuestion("Axample.com", T_A)); |
|
data = pkt.data(); |
|
EXPECT_EQ(ARES_SUCCESS, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
ASSERT_NE(nullptr, host); |
|
std::stringstream ss; |
|
ss << HostEnt(host); |
|
EXPECT_EQ("{'Axample.com' aliases=[] addrs=[2.3.4.5]}", ss.str()); |
|
ares_free_hostent(host); |
|
host = nullptr; |
|
|
|
pkt.questions_.clear(); |
|
pkt.add_question(new DNSQuestion("example.com", T_A)); |
|
|
|
#ifdef DISABLED |
|
// Not a response. |
|
pkt.set_response(false); |
|
data = pkt.data(); |
|
EXPECT_EQ(ARES_EBADRESP, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
EXPECT_EQ(nullptr, host); |
|
pkt.set_response(true); |
|
|
|
// Bad return code. |
|
pkt.set_rcode(FORMERR); |
|
data = pkt.data(); |
|
EXPECT_EQ(ARES_ENODATA, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
EXPECT_EQ(nullptr, host); |
|
pkt.set_rcode(NOERROR); |
|
#endif |
|
|
|
// Two questions |
|
pkt.add_question(new DNSQuestion("example.com", T_A)); |
|
data = pkt.data(); |
|
EXPECT_EQ(ARES_EBADRESP, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
EXPECT_EQ(nullptr, host); |
|
pkt.questions_.clear(); |
|
pkt.add_question(new DNSQuestion("example.com", T_A)); |
|
|
|
// 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_ENODATA, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
EXPECT_EQ(nullptr, host); |
|
pkt.answers_.clear(); |
|
pkt.add_answer(new DNSARR("example.com", 100, {0x02, 0x03, 0x04, 0x05})); |
|
|
|
// No answer. |
|
pkt.answers_.clear(); |
|
data = pkt.data(); |
|
EXPECT_EQ(ARES_ENODATA, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)); |
|
EXPECT_EQ(nullptr, host); |
|
pkt.add_answer(new DNSARR("example.com", 100, {0x02, 0x03, 0x04, 0x05})); |
|
|
|
// Truncated packets. |
|
data = pkt.data(); |
|
for (size_t len = 1; len < data.size(); len++) { |
|
EXPECT_EQ(ARES_EBADRESP, ares_parse_a_reply(data.data(), (int)len, |
|
&host, info, &count)); |
|
EXPECT_EQ(nullptr, host); |
|
EXPECT_EQ(ARES_EBADRESP, ares_parse_a_reply(data.data(), (int)len, |
|
nullptr, info, &count)); |
|
} |
|
} |
|
|
|
TEST_F(LibraryTest, ParseAReplyAllocFail) { |
|
DNSPacket pkt; |
|
pkt.set_qid(0x1234).set_response().set_aa() |
|
.add_question(new DNSQuestion("example.com", T_A)) |
|
.add_answer(new DNSCnameRR("example.com", 300, "c.example.com")) |
|
.add_answer(new DNSARR("c.example.com", 500, {0x02, 0x03, 0x04, 0x05})); |
|
std::vector<byte> data = pkt.data(); |
|
|
|
struct hostent *host = nullptr; |
|
struct ares_addrttl info[2]; |
|
int count = 2; |
|
|
|
for (int ii = 1; ii <= 8; ii++) { |
|
ClearFails(); |
|
SetAllocFail(ii); |
|
EXPECT_EQ(ARES_ENOMEM, ares_parse_a_reply(data.data(), (int)data.size(), |
|
&host, info, &count)) << ii; |
|
EXPECT_EQ(nullptr, host); |
|
} |
|
} |
|
|
|
} // namespace test |
|
} // namespace ares
|
|
|