mirror of https://github.com/c-ares/c-ares.git
Add support for URI(Uniform Resource Identifier) records. (#411)
Add ares_parse_uri_reply() for parsing URI DNS replies. Fix By: Martin Holeš (@martin-256)pull/414/head
parent
ccebec8303
commit
9ac7efc382
16 changed files with 647 additions and 7 deletions
@ -0,0 +1,81 @@ |
||||
.\" |
||||
.\" Copyright 1998 by the Massachusetts Institute of Technology. |
||||
.\" |
||||
.\" Permission to use, copy, modify, and distribute this |
||||
.\" software and its documentation for any purpose and without |
||||
.\" fee is hereby granted, provided that the above copyright |
||||
.\" notice appear in all copies and that both that copyright |
||||
.\" notice and this permission notice appear in supporting |
||||
.\" documentation, and that the name of M.I.T. not be used in |
||||
.\" advertising or publicity pertaining to distribution of the |
||||
.\" software without specific, written prior permission. |
||||
.\" M.I.T. makes no representations about the suitability of |
||||
.\" this software for any purpose. It is provided "as is" |
||||
.\" without express or implied warranty. |
||||
.\" |
||||
.TH ARES_PARSE_URI_REPLY 3 "14 August 2020" |
||||
.SH NAME |
||||
ares_parse_uri_reply \- Parse a reply to a DNS query of type URI |
||||
.SH SYNOPSIS |
||||
.nf |
||||
.B #include <ares.h> |
||||
.PP |
||||
.B int ares_parse_uri_reply(const unsigned char* \fIabuf\fP, int \fIalen\fP, |
||||
.B struct ares_uri_reply** \fIuri_out\fP); |
||||
.fi |
||||
.SH DESCRIPTION |
||||
The |
||||
.B ares_parse_uri_reply |
||||
function parses the response to a query of type URI into a |
||||
linked list of |
||||
.I struct ares_uri_reply |
||||
The parameters |
||||
.I abuf |
||||
and |
||||
.I alen |
||||
give the contents of the response. The result is stored in allocated |
||||
memory and a pointer to it stored into the variable pointed to by |
||||
.IR uri_out . |
||||
It is the caller's responsibility to free the resulting |
||||
.IR uri_out |
||||
structure when it is no longer needed using the function |
||||
.B ares_free_data |
||||
.PP |
||||
The structure |
||||
.I ares_uri_reply |
||||
contains the following fields: |
||||
.sp |
||||
.in +4n |
||||
.nf |
||||
struct ares_uri_reply { |
||||
struct ares_uri_reply *next; |
||||
unsigned short weight; |
||||
unsigned short priority; |
||||
char *uri; |
||||
int ttl; |
||||
}; |
||||
.fi |
||||
.in |
||||
.PP |
||||
.SH RETURN VALUES |
||||
.B ares_parse_uri_reply |
||||
can return any of the following values: |
||||
.TP 15 |
||||
.B ARES_SUCCESS |
||||
The response was successfully parsed. |
||||
.TP 15 |
||||
.B ARES_EBADRESP |
||||
The response was malformatted. |
||||
.TP 15 |
||||
.B ARES_ENODATA |
||||
The response did not contain an answer to the query. |
||||
.TP 15 |
||||
.B ARES_ENOMEM |
||||
Memory was exhausted. |
||||
.SH AVAILABILITY |
||||
|
||||
.SH SEE ALSO |
||||
.BR ares_query (3) |
||||
.BR ares_free_data (3) |
||||
.SH AUTHOR |
||||
Written by Jan Petrasek <petrasek@tes.eu> |
@ -0,0 +1,184 @@ |
||||
|
||||
/* Copyright 1998 by the Massachusetts Institute of Technology.
|
||||
* Copyright (C) 2009 by Jakub Hrozek <jhrozek@redhat.com> |
||||
* |
||||
* Permission to use, copy, modify, and distribute this |
||||
* software and its documentation for any purpose and without |
||||
* fee is hereby granted, provided that the above copyright |
||||
* notice appear in all copies and that both that copyright |
||||
* notice and this permission notice appear in supporting |
||||
* documentation, and that the name of M.I.T. not be used in |
||||
* advertising or publicity pertaining to distribution of the |
||||
* software without specific, written prior permission. |
||||
* M.I.T. makes no representations about the suitability of |
||||
* this software for any purpose. It is provided "as is" |
||||
* without express or implied warranty. |
||||
*/ |
||||
|
||||
#include "ares_setup.h" |
||||
|
||||
#ifdef HAVE_NETINET_IN_H |
||||
# include <netinet/in.h> |
||||
#endif |
||||
#ifdef HAVE_NETDB_H |
||||
# include <netdb.h> |
||||
#endif |
||||
#ifdef HAVE_ARPA_INET_H |
||||
# include <arpa/inet.h> |
||||
#endif |
||||
|
||||
#include "ares_nameser.h" |
||||
|
||||
#include "ares.h" |
||||
#include "ares_dns.h" |
||||
#include "ares_data.h" |
||||
#include "ares_private.h" |
||||
|
||||
/* AIX portability check */ |
||||
#ifndef T_URI |
||||
# define T_URI 256 /* uri selection */ |
||||
#endif |
||||
|
||||
int |
||||
ares_parse_uri_reply (const unsigned char *abuf, int alen, |
||||
struct ares_uri_reply **uri_out) |
||||
{ |
||||
unsigned int qdcount, ancount, i; |
||||
const unsigned char *aptr, *vptr; |
||||
int status, rr_type, rr_class, rr_len, rr_ttl; |
||||
long len; |
||||
char *uri_str = NULL, *rr_name = NULL; |
||||
struct ares_uri_reply *uri_head = NULL; |
||||
struct ares_uri_reply *uri_last = NULL; |
||||
struct ares_uri_reply *uri_curr; |
||||
|
||||
/* Set *uri_out to NULL for all failure cases. */ |
||||
*uri_out = NULL; |
||||
|
||||
/* Give up if abuf doesn't have room for a header. */ |
||||
if (alen < HFIXEDSZ){ |
||||
return ARES_EBADRESP; |
||||
} |
||||
|
||||
/* Fetch the question and answer count from the header. */ |
||||
qdcount = DNS_HEADER_QDCOUNT (abuf); |
||||
ancount = DNS_HEADER_ANCOUNT (abuf); |
||||
if (qdcount != 1) { |
||||
return ARES_EBADRESP; |
||||
} |
||||
if (ancount == 0) { |
||||
return ARES_ENODATA; |
||||
} |
||||
/* Expand the name from the question, and skip past the question. */ |
||||
aptr = abuf + HFIXEDSZ; |
||||
|
||||
status = ares_expand_name (aptr, abuf, alen, &uri_str, &len); |
||||
if (status != ARES_SUCCESS){ |
||||
return status; |
||||
} |
||||
if (aptr + len + QFIXEDSZ > abuf + alen) |
||||
{ |
||||
ares_free (uri_str); |
||||
return ARES_EBADRESP; |
||||
} |
||||
aptr += len + QFIXEDSZ; |
||||
|
||||
/* Examine each answer resource record (RR) in turn. */ |
||||
for (i = 0; i < ancount; i++) |
||||
{ |
||||
/* Decode the RR up to the data field. */ |
||||
status = ares_expand_name (aptr, abuf, alen, &rr_name, &len); |
||||
if (status != ARES_SUCCESS) |
||||
{ |
||||
break; |
||||
} |
||||
aptr += len; |
||||
if (aptr + RRFIXEDSZ > abuf + alen) |
||||
{ |
||||
status = ARES_EBADRESP; |
||||
break; |
||||
} |
||||
|
||||
rr_type = DNS_RR_TYPE (aptr); |
||||
rr_class = DNS_RR_CLASS (aptr); |
||||
rr_ttl = DNS_RR_TTL(aptr); |
||||
rr_len = DNS_RR_LEN (aptr); |
||||
aptr += RRFIXEDSZ; |
||||
|
||||
if (aptr + rr_len > abuf + alen) |
||||
{ |
||||
status = ARES_EBADRESP; |
||||
break; |
||||
} |
||||
|
||||
/* Check if we are really looking at a URI record */ |
||||
if (rr_class == C_IN && rr_type == T_URI) |
||||
{ |
||||
/* parse the URI record itself */ |
||||
if (rr_len < 5) |
||||
{ |
||||
status = ARES_EBADRESP; |
||||
break; |
||||
} |
||||
/* Allocate storage for this URI answer appending it to the list */ |
||||
uri_curr = ares_malloc_data(ARES_DATATYPE_URI_REPLY); |
||||
if (!uri_curr) |
||||
{ |
||||
status = ARES_ENOMEM; |
||||
break; |
||||
} |
||||
if (uri_last) |
||||
{ |
||||
uri_last->next = uri_curr; |
||||
} |
||||
else |
||||
{ |
||||
uri_head = uri_curr; |
||||
} |
||||
uri_last = uri_curr; |
||||
|
||||
vptr = aptr; |
||||
uri_curr->priority = DNS__16BIT(vptr); |
||||
vptr += sizeof(unsigned short); |
||||
uri_curr->weight = DNS__16BIT(vptr); |
||||
vptr += sizeof(unsigned short); |
||||
uri_curr->uri = (char *)ares_malloc(rr_len-3); |
||||
if (!uri_curr->uri) |
||||
{ |
||||
status = ARES_ENOMEM; |
||||
break; |
||||
} |
||||
uri_curr->uri = strncpy(uri_curr->uri, vptr, rr_len-4); |
||||
uri_curr->uri[rr_len-4]='\0'; |
||||
uri_curr->ttl = rr_ttl; |
||||
|
||||
if (status != ARES_SUCCESS) |
||||
break; |
||||
} |
||||
|
||||
/* Don't lose memory in the next iteration */ |
||||
ares_free (rr_name); |
||||
rr_name = NULL; |
||||
|
||||
/* Move on to the next record */ |
||||
aptr += rr_len; |
||||
} |
||||
|
||||
if (uri_str) |
||||
ares_free (uri_str); |
||||
if (rr_name) |
||||
ares_free (rr_name); |
||||
|
||||
/* clean up on error */ |
||||
if (status != ARES_SUCCESS) |
||||
{ |
||||
if (uri_head) |
||||
ares_free_data (uri_head); |
||||
return status; |
||||
} |
||||
|
||||
/* everything looks fine, return the data */ |
||||
*uri_out = uri_head; |
||||
|
||||
return ARES_SUCCESS; |
||||
} |
@ -0,0 +1,288 @@ |
||||
#include "ares-test.h" |
||||
#include "dns-proto.h" |
||||
|
||||
#include <sstream> |
||||
#include <vector> |
||||
|
||||
namespace ares { |
||||
namespace test { |
||||
|
||||
TEST_F(LibraryTest, ParseUriReplyOK) { |
||||
DNSPacket pkt; |
||||
pkt.set_qid(0x1234).set_response().set_aa() |
||||
.add_question(new DNSQuestion("example.com", T_URI)) |
||||
.add_answer(new DNSUriRR("example.com", 100, 10, 20, "uri.example.com")) |
||||
.add_answer(new DNSUriRR("example.com", 200, 11, 21, "uri2.example.com")); |
||||
std::vector<byte> data = pkt.data(); |
||||
|
||||
struct ares_uri_reply* uri = nullptr; |
||||
EXPECT_EQ(ARES_SUCCESS, ares_parse_uri_reply(data.data(), data.size(), &uri)); |
||||
ASSERT_NE(nullptr, uri); |
||||
|
||||
EXPECT_EQ("uri.example.com", std::string(uri->uri)); |
||||
EXPECT_EQ(10, uri->priority); |
||||
EXPECT_EQ(20, uri->weight); |
||||
EXPECT_EQ(100, uri->ttl); |
||||
|
||||
struct ares_uri_reply* uri2 = uri->next; |
||||
ASSERT_NE(nullptr, uri2); |
||||
EXPECT_EQ("uri2.example.com", std::string(uri2->uri)); |
||||
EXPECT_EQ(11, uri2->priority); |
||||
EXPECT_EQ(21, uri2->weight); |
||||
EXPECT_EQ(200, uri2->ttl); |
||||
EXPECT_EQ(nullptr, uri2->next); |
||||
|
||||
ares_free_data(uri); |
||||
} |
||||
|
||||
TEST_F(LibraryTest, ParseUriReplySingle) { |
||||
DNSPacket pkt; |
||||
pkt.set_qid(0x1234).set_response().set_aa() |
||||
.add_question(new DNSQuestion("example.abc.def.com", T_URI)) |
||||
.add_answer(new DNSUriRR("example.abc.def.com", 180, 0, 10, "example.abc.def.com")) |
||||
.add_auth(new DNSNsRR("abc.def.com", 44, "else1.where.com")) |
||||
.add_auth(new DNSNsRR("abc.def.com", 44, "else2.where.com")) |
||||
.add_auth(new DNSNsRR("abc.def.com", 44, "else3.where.com")) |
||||
.add_auth(new DNSNsRR("abc.def.com", 44, "else4.where.com")) |
||||
.add_auth(new DNSNsRR("abc.def.com", 44, "else5.where.com")) |
||||
.add_additional(new DNSARR("else2.where.com", 42, {172,19,0,1})) |
||||
.add_additional(new DNSARR("else5.where.com", 42, {172,19,0,2})); |
||||
std::vector<byte> data = pkt.data(); |
||||
|
||||
struct ares_uri_reply* uri = nullptr; |
||||
EXPECT_EQ(ARES_SUCCESS, ares_parse_uri_reply(data.data(), data.size(), &uri)); |
||||
ASSERT_NE(nullptr, uri); |
||||
|
||||
EXPECT_EQ("example.abc.def.com", std::string(uri->uri)); |
||||
EXPECT_EQ(0, uri->priority); |
||||
EXPECT_EQ(10, uri->weight); |
||||
EXPECT_EQ(180, uri->ttl); |
||||
EXPECT_EQ(nullptr, uri->next); |
||||
|
||||
ares_free_data(uri); |
||||
} |
||||
|
||||
TEST_F(LibraryTest, ParseUriReplyMalformed) { |
||||
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, 0x01, // 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, |
||||
0x01, 0x00, // type URI
|
||||
0x00, 0x01, // class IN
|
||||
// Answer 1
|
||||
0x07, 'e', 'x', 'a', 'm', 'p', 'l', 'e', |
||||
0x03, 'c', 'o', 'm', |
||||
0x00, |
||||
0x01, 0x00, // RR type
|
||||
0x00, 0x01, // class IN
|
||||
0x01, 0x02, 0x03, 0x04, // TTL
|
||||
0x00, 0x04, // rdata length -- too short
|
||||
0x02, 0x03, 0x04, 0x05, |
||||
}; |
||||
|
||||
struct ares_uri_reply* uri = nullptr; |
||||
EXPECT_EQ(ARES_EBADRESP, ares_parse_uri_reply(data.data(), data.size(), &uri)); |
||||
ASSERT_EQ(nullptr, uri); |
||||
} |
||||
|
||||
TEST_F(LibraryTest, ParseUriReplyMultiple) { |
||||
DNSPacket pkt; |
||||
pkt.set_qid(0x1234).set_response().set_ra().set_rd() |
||||
.add_question(new DNSQuestion("uri.example.com", T_URI)) |
||||
.add_answer(new DNSUriRR("uri.example.com", 600, 0, 5, "a1.uri.example.com")) |
||||
.add_answer(new DNSUriRR("uri.example.com", 660, 0, 5, "a2.uri.example.com")) |
||||
.add_answer(new DNSUriRR("uri.example.com", 720, 0, 5, "a3.uri.example.com")) |
||||
.add_auth(new DNSNsRR("example.com", 300, "ns1.example.com")) |
||||
.add_auth(new DNSNsRR("example.com", 300, "ns2.example.com")) |
||||
.add_auth(new DNSNsRR("example.com", 300, "ns3.example.com")) |
||||
.add_additional(new DNSARR("a1.uri.example.com", 300, {172,19,1,1})) |
||||
.add_additional(new DNSARR("a2.uri.example.com", 300, {172,19,1,2})) |
||||
.add_additional(new DNSARR("a3.uri.example.com", 300, {172,19,1,3})) |
||||
.add_additional(new DNSARR("n1.example.com", 300, {172,19,0,1})) |
||||
.add_additional(new DNSARR("n2.example.com", 300, {172,19,0,2})) |
||||
.add_additional(new DNSARR("n3.example.com", 300, {172,19,0,3})); |
||||
std::vector<byte> data = pkt.data(); |
||||
|
||||
struct ares_uri_reply* uri0 = nullptr; |
||||
EXPECT_EQ(ARES_SUCCESS, ares_parse_uri_reply(data.data(), data.size(), &uri0)); |
||||
ASSERT_NE(nullptr, uri0); |
||||
struct ares_uri_reply* uri = uri0; |
||||
|
||||
EXPECT_EQ("a1.uri.example.com", std::string(uri->uri)); |
||||
EXPECT_EQ(0, uri->priority); |
||||
EXPECT_EQ(5, uri->weight); |
||||
EXPECT_EQ(600, uri->ttl); |
||||
EXPECT_NE(nullptr, uri->next); |
||||
uri = uri->next; |
||||
|
||||
EXPECT_EQ("a2.uri.example.com", std::string(uri->uri)); |
||||
EXPECT_EQ(0, uri->priority); |
||||
EXPECT_EQ(5, uri->weight); |
||||
EXPECT_EQ(660, uri->ttl); |
||||
EXPECT_NE(nullptr, uri->next); |
||||
uri = uri->next; |
||||
|
||||
EXPECT_EQ("a3.uri.example.com", std::string(uri->uri)); |
||||
EXPECT_EQ(0, uri->priority); |
||||
EXPECT_EQ(5, uri->weight); |
||||
EXPECT_EQ(720, uri->ttl); |
||||
EXPECT_EQ(nullptr, uri->next); |
||||
|
||||
ares_free_data(uri0); |
||||
} |
||||
|
||||
TEST_F(LibraryTest, ParseUriReplyCname) { |
||||
DNSPacket pkt; |
||||
pkt.set_qid(0x1234).set_response().set_aa() |
||||
.add_question(new DNSQuestion("example.abc.def.com", T_URI)) |
||||
.add_answer(new DNSCnameRR("example.abc.def.com", 300, "cname.abc.def.com")) |
||||
.add_answer(new DNSUriRR("cname.abc.def.com", 600, 0, 10, "uri.abc.def.com")) |
||||
.add_auth(new DNSNsRR("abc.def.com", 44, "else1.where.com")) |
||||
.add_auth(new DNSNsRR("abc.def.com", 44, "else2.where.com")) |
||||
.add_auth(new DNSNsRR("abc.def.com", 44, "else3.where.com")) |
||||
.add_additional(new DNSARR("example.abc.def.com", 300, {172,19,0,1})) |
||||
.add_additional(new DNSARR("else1.where.com", 42, {172,19,0,1})) |
||||
.add_additional(new DNSARR("else2.where.com", 42, {172,19,0,2})) |
||||
.add_additional(new DNSARR("else3.where.com", 42, {172,19,0,3})); |
||||
std::vector<byte> data = pkt.data(); |
||||
|
||||
struct ares_uri_reply* uri = nullptr; |
||||
EXPECT_EQ(ARES_SUCCESS, ares_parse_uri_reply(data.data(), data.size(), &uri)); |
||||
ASSERT_NE(nullptr, uri); |
||||
|
||||
EXPECT_EQ("uri.abc.def.com", std::string(uri->uri)); |
||||
EXPECT_EQ(0, uri->priority); |
||||
EXPECT_EQ(10, uri->weight); |
||||
EXPECT_EQ(600, uri->ttl); |
||||
EXPECT_EQ(nullptr, uri->next); |
||||
|
||||
ares_free_data(uri); |
||||
} |
||||
|
||||
TEST_F(LibraryTest, ParseUriReplyCnameMultiple) { |
||||
DNSPacket pkt; |
||||
pkt.set_qid(0x1234).set_response().set_ra().set_rd() |
||||
.add_question(new DNSQuestion("query.example.com", T_URI)) |
||||
.add_answer(new DNSCnameRR("query.example.com", 300, "uri.example.com")) |
||||
.add_answer(new DNSUriRR("uri.example.com", 600, 0, 5, "a1.uri.example.com")) |
||||
.add_answer(new DNSUriRR("uri.example.com", 660, 0, 5, "a2.uri.example.com")) |
||||
.add_answer(new DNSUriRR("uri.example.com", 720, 0, 5, "a3.uri.example.com")) |
||||
.add_auth(new DNSNsRR("example.com", 300, "ns1.example.com")) |
||||
.add_auth(new DNSNsRR("example.com", 300, "ns2.example.com")) |
||||
.add_auth(new DNSNsRR("example.com", 300, "ns3.example.com")) |
||||
.add_additional(new DNSARR("a1.uri.example.com", 300, {172,19,1,1})) |
||||
.add_additional(new DNSARR("a2.uri.example.com", 300, {172,19,1,2})) |
||||
.add_additional(new DNSARR("a3.uri.example.com", 300, {172,19,1,3})) |
||||
.add_additional(new DNSARR("n1.example.com", 300, {172,19,0,1})) |
||||
.add_additional(new DNSARR("n2.example.com", 300, {172,19,0,2})) |
||||
.add_additional(new DNSARR("n3.example.com", 300, {172,19,0,3})); |
||||
std::vector<byte> data = pkt.data(); |
||||
|
||||
struct ares_uri_reply* uri0 = nullptr; |
||||
EXPECT_EQ(ARES_SUCCESS, ares_parse_uri_reply(data.data(), data.size(), &uri0)); |
||||
ASSERT_NE(nullptr, uri0); |
||||
struct ares_uri_reply* uri = uri0; |
||||
|
||||
EXPECT_EQ("a1.uri.example.com", std::string(uri->uri)); |
||||
EXPECT_EQ(0, uri->priority); |
||||
EXPECT_EQ(5, uri->weight); |
||||
EXPECT_EQ(600, uri->ttl); |
||||
EXPECT_NE(nullptr, uri->next); |
||||
uri = uri->next; |
||||
|
||||
EXPECT_EQ("a2.uri.example.com", std::string(uri->uri)); |
||||
EXPECT_EQ(0, uri->priority); |
||||
EXPECT_EQ(5, uri->weight); |
||||
EXPECT_EQ(660, uri->ttl); |
||||
EXPECT_NE(nullptr, uri->next); |
||||
uri = uri->next; |
||||
|
||||
EXPECT_EQ("a3.uri.example.com", std::string(uri->uri)); |
||||
EXPECT_EQ(0, uri->priority); |
||||
EXPECT_EQ(5, uri->weight); |
||||
EXPECT_EQ(720, uri->ttl); |
||||
EXPECT_EQ(nullptr, uri->next); |
||||
|
||||
ares_free_data(uri0); |
||||
} |
||||
|
||||
TEST_F(LibraryTest, ParseUriReplyErrors) { |
||||
DNSPacket pkt; |
||||
pkt.set_qid(0x1234).set_response().set_aa() |
||||
.add_question(new DNSQuestion("example.abc.def.com", T_URI)) |
||||
.add_answer(new DNSUriRR("example.abc.def.com", 180, 0, 10, "example.abc.def.com")); |
||||
std::vector<byte> data; |
||||
struct ares_uri_reply* uri = nullptr; |
||||
|
||||
// No question.
|
||||
pkt.questions_.clear(); |
||||
data = pkt.data(); |
||||
EXPECT_EQ(ARES_EBADRESP, ares_parse_uri_reply(data.data(), data.size(), &uri)); |
||||
pkt.add_question(new DNSQuestion("example.abc.def.com", T_URI)); |
||||
|
||||
#ifdef DISABLED |
||||
// Question != answer
|
||||
pkt.questions_.clear(); |
||||
pkt.add_question(new DNSQuestion("Axample.com", T_URI)); |
||||
data = pkt.data(); |
||||
EXPECT_EQ(ARES_ENODATA, ares_parse_uri_reply(data.data(), data.size(), &uri)); |
||||
pkt.questions_.clear(); |
||||
pkt.add_question(new DNSQuestion("example.com", T_URI)); |
||||
#endif |
||||
|
||||
// Two questions.
|
||||
pkt.add_question(new DNSQuestion("example.abc.def.com", T_URI)); |
||||
data = pkt.data(); |
||||
EXPECT_EQ(ARES_EBADRESP, ares_parse_uri_reply(data.data(), data.size(), &uri)); |
||||
pkt.questions_.clear(); |
||||
pkt.add_question(new DNSQuestion("64.48.32.16.in-addr.arpa", T_PTR)); |
||||
|
||||
// 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_SUCCESS, ares_parse_uri_reply(data.data(), data.size(), &uri)); |
||||
EXPECT_EQ(nullptr, uri); |
||||
pkt.answers_.clear(); |
||||
pkt.add_answer(new DNSUriRR("example.abc.def.com", 180, 0, 10, "example.abc.def.com")); |
||||
|
||||
// No answer.
|
||||
pkt.answers_.clear(); |
||||
data = pkt.data(); |
||||
EXPECT_EQ(ARES_ENODATA, ares_parse_uri_reply(data.data(), data.size(), &uri)); |
||||
pkt.add_answer(new DNSUriRR("example.abc.def.com", 180, 0, 10, "example.abc.def.com")); |
||||
|
||||
// Truncated packets.
|
||||
data = pkt.data(); |
||||
for (size_t len = 1; len < data.size(); len++) { |
||||
int rc = ares_parse_uri_reply(data.data(), len, &uri); |
||||
EXPECT_TRUE(rc == ARES_EBADRESP || rc == ARES_EBADNAME); |
||||
} |
||||
} |
||||
|
||||
TEST_F(LibraryTest, ParseUriReplyAllocFail) { |
||||
DNSPacket pkt; |
||||
pkt.set_qid(0x1234).set_response().set_aa() |
||||
.add_question(new DNSQuestion("example.abc.def.com", T_URI)) |
||||
.add_answer(new DNSCnameRR("example.com", 300, "c.example.com")) |
||||
.add_answer(new DNSUriRR("example.abc.def.com", 180, 0, 10, "example.abc.def.com")); |
||||
std::vector<byte> data = pkt.data(); |
||||
struct ares_uri_reply* uri = nullptr; |
||||
|
||||
for (int ii = 1; ii <= 5; ii++) { |
||||
ClearFails(); |
||||
SetAllocFail(ii); |
||||
EXPECT_EQ(ARES_ENOMEM, ares_parse_uri_reply(data.data(), data.size(), &uri)) << ii; |
||||
} |
||||
} |
||||
|
||||
} // namespace test
|
||||
} // namespace ares
|
Loading…
Reference in new issue