|
|
|
/* 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"
|
|
|
|
|
|
|
|
#ifdef CARES_THREADS
|
|
|
|
|
|
|
|
#ifndef WIN32
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <sstream>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
using testing::InvokeWithoutArgs;
|
|
|
|
using testing::DoAll;
|
|
|
|
|
|
|
|
namespace ares {
|
|
|
|
namespace test {
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, Basic) {
|
|
|
|
std::vector<byte> reply = {
|
|
|
|
0x00, 0x00, // qid
|
|
|
|
0x84, // response + query + AA + not-TC + not-RD
|
|
|
|
0x00, // not-RA + not-Z + not-AD + not-CD + rc=NoError
|
|
|
|
0x00, 0x01, // 1 question
|
|
|
|
0x00, 0x01, // 1 answer RRs
|
|
|
|
0x00, 0x00, // 0 authority RRs
|
|
|
|
0x00, 0x00, // 0 additional RRs
|
|
|
|
// Question
|
|
|
|
0x03, 'w', 'w', 'w',
|
|
|
|
0x06, 'g', 'o', 'o', 'g', 'l', 'e',
|
|
|
|
0x03, 'c', 'o', 'm',
|
|
|
|
0x00,
|
|
|
|
0x00, 0x01, // type A
|
|
|
|
0x00, 0x01, // class IN
|
|
|
|
// Answer
|
|
|
|
0x03, 'w', 'w', 'w',
|
|
|
|
0x06, 'g', 'o', 'o', 'g', 'l', 'e',
|
|
|
|
0x03, 'c', 'o', 'm',
|
|
|
|
0x00,
|
|
|
|
0x00, 0x01, // type A
|
|
|
|
0x00, 0x01, // class IN
|
|
|
|
0x00, 0x00, 0x01, 0x00, // TTL
|
|
|
|
0x00, 0x04, // rdata length
|
|
|
|
0x01, 0x02, 0x03, 0x04
|
|
|
|
};
|
|
|
|
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReplyData(&server_, reply));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[1.2.3.4]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
// UDP only so mock server doesn't get confused by concatenated requests
|
|
|
|
TEST_P(MockUDPEventThreadTest, GetHostByNameParallelLookups) {
|
|
|
|
DNSPacket rsp1;
|
|
|
|
rsp1.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 100, {2, 3, 4, 5}));
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp1));
|
|
|
|
DNSPacket rsp2;
|
|
|
|
rsp2.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.example.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.example.com", 100, {1, 2, 3, 4}));
|
|
|
|
ON_CALL(server_, OnRequest("www.example.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp2));
|
|
|
|
|
|
|
|
HostResult result1;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result1);
|
|
|
|
HostResult result2;
|
|
|
|
ares_gethostbyname(channel_, "www.example.com.", AF_INET, HostCallback, &result2);
|
|
|
|
HostResult result3;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result3);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result1.done_);
|
|
|
|
EXPECT_TRUE(result2.done_);
|
|
|
|
EXPECT_TRUE(result3.done_);
|
|
|
|
std::stringstream ss1;
|
|
|
|
ss1 << result1.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[2.3.4.5]}", ss1.str());
|
|
|
|
std::stringstream ss2;
|
|
|
|
ss2 << result2.host_;
|
|
|
|
EXPECT_EQ("{'www.example.com' aliases=[] addrs=[1.2.3.4]}", ss2.str());
|
|
|
|
std::stringstream ss3;
|
|
|
|
ss3 << result3.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[2.3.4.5]}", ss3.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
// UDP to TCP specific test
|
|
|
|
TEST_P(MockUDPEventThreadTest, TruncationRetry) {
|
|
|
|
DNSPacket rsptruncated;
|
|
|
|
rsptruncated.set_response().set_aa().set_tc()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A));
|
|
|
|
DNSPacket rspok;
|
|
|
|
rspok.set_response()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 100, {1, 2, 3, 4}));
|
|
|
|
EXPECT_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillOnce(SetReply(&server_, &rsptruncated))
|
|
|
|
.WillOnce(SetReply(&server_, &rspok));
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[1.2.3.4]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
static int sock_cb_count = 0;
|
|
|
|
static int SocketConnectCallback(ares_socket_t fd, int type, void *data) {
|
|
|
|
int rc = *(int*)data;
|
|
|
|
if (verbose) std::cerr << "SocketConnectCallback(" << fd << ") invoked" << std::endl;
|
|
|
|
sock_cb_count++;
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, SockCallback) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 100, {2, 3, 4, 5}));
|
|
|
|
EXPECT_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillOnce(SetReply(&server_, &rsp));
|
|
|
|
|
|
|
|
// Get notified of new sockets
|
|
|
|
int rc = ARES_SUCCESS;
|
|
|
|
ares_set_socket_callback(channel_, SocketConnectCallback, &rc);
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
sock_cb_count = 0;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_EQ(1, sock_cb_count);
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, SockFailCallback) {
|
|
|
|
// Notification of new sockets gives an error.
|
|
|
|
int rc = -1;
|
|
|
|
ares_set_socket_callback(channel_, SocketConnectCallback, &rc);
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
sock_cb_count = 0;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_LT(1, sock_cb_count);
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ECONNREFUSED, result.status_);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, ReInit) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 100, {2, 3, 4, 5}));
|
|
|
|
EXPECT_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillOnce(SetReply(&server_, &rsp));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
EXPECT_EQ(ARES_SUCCESS, ares_reinit(channel_));
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
#define MAXUDPQUERIES_TOTAL 32
|
|
|
|
#define MAXUDPQUERIES_LIMIT 8
|
|
|
|
|
|
|
|
class MockUDPEventThreadMaxQueriesTest
|
|
|
|
: public MockEventThreadOptsTest,
|
|
|
|
public ::testing::WithParamInterface<std::tuple<ares_evsys_t,int>> {
|
|
|
|
public:
|
|
|
|
MockUDPEventThreadMaxQueriesTest()
|
|
|
|
: MockEventThreadOptsTest(1, std::get<0>(GetParam()), std::get<1>(GetParam()), false,
|
|
|
|
FillOptions(&opts_),
|
|
|
|
ARES_OPT_UDP_MAX_QUERIES|ARES_OPT_FLAGS) {}
|
|
|
|
static struct ares_options* FillOptions(struct ares_options * opts) {
|
|
|
|
memset(opts, 0, sizeof(struct ares_options));
|
|
|
|
opts->flags = ARES_FLAG_STAYOPEN|ARES_FLAG_EDNS;
|
|
|
|
opts->udp_max_queries = MAXUDPQUERIES_LIMIT;
|
|
|
|
return opts;
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
struct ares_options opts_;
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_P(MockUDPEventThreadMaxQueriesTest, GetHostByNameParallelLookups) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 100, {2, 3, 4, 5}));
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp));
|
|
|
|
|
|
|
|
// Get notified of new sockets so we can validate how many are created
|
|
|
|
int rc = ARES_SUCCESS;
|
|
|
|
ares_set_socket_callback(channel_, SocketConnectCallback, &rc);
|
|
|
|
sock_cb_count = 0;
|
|
|
|
|
|
|
|
HostResult result[MAXUDPQUERIES_TOTAL];
|
|
|
|
for (size_t i=0; i<MAXUDPQUERIES_TOTAL; i++) {
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
Process();
|
|
|
|
|
|
|
|
EXPECT_EQ(MAXUDPQUERIES_TOTAL / MAXUDPQUERIES_LIMIT, sock_cb_count);
|
|
|
|
|
|
|
|
for (size_t i=0; i<MAXUDPQUERIES_TOTAL; i++) {
|
|
|
|
std::stringstream ss;
|
|
|
|
EXPECT_TRUE(result[i].done_);
|
|
|
|
ss << result[i].host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class CacheQueriesEventThreadTest
|
|
|
|
: public MockEventThreadOptsTest,
|
|
|
|
public ::testing::WithParamInterface<std::tuple<ares_evsys_t,int>> {
|
|
|
|
public:
|
|
|
|
CacheQueriesEventThreadTest()
|
|
|
|
: MockEventThreadOptsTest(1, std::get<0>(GetParam()), std::get<1>(GetParam()), false,
|
|
|
|
FillOptions(&opts_),
|
|
|
|
ARES_OPT_QUERY_CACHE) {}
|
|
|
|
static struct ares_options* FillOptions(struct ares_options * opts) {
|
|
|
|
memset(opts, 0, sizeof(struct ares_options));
|
|
|
|
opts->qcache_max_ttl = 3600;
|
|
|
|
return opts;
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
struct ares_options opts_;
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_P(CacheQueriesEventThreadTest, GetHostByNameCache) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 100, {2, 3, 4, 5}));
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp));
|
|
|
|
|
|
|
|
// Get notified of new sockets so we can validate how many are created
|
|
|
|
int rc = ARES_SUCCESS;
|
|
|
|
ares_set_socket_callback(channel_, SocketConnectCallback, &rc);
|
|
|
|
sock_cb_count = 0;
|
|
|
|
|
|
|
|
HostResult result1;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result1);
|
|
|
|
Process();
|
|
|
|
|
|
|
|
std::stringstream ss1;
|
|
|
|
EXPECT_TRUE(result1.done_);
|
|
|
|
ss1 << result1.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[2.3.4.5]}", ss1.str());
|
|
|
|
|
|
|
|
/* Run again, should return cached result */
|
|
|
|
HostResult result2;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result2);
|
|
|
|
Process();
|
|
|
|
|
|
|
|
std::stringstream ss2;
|
|
|
|
EXPECT_TRUE(result2.done_);
|
|
|
|
ss2 << result2.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[2.3.4.5]}", ss2.str());
|
|
|
|
|
|
|
|
EXPECT_EQ(1, sock_cb_count);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define TCPPARALLELLOOKUPS 32
|
|
|
|
|
|
|
|
class MockTCPEventThreadStayOpenTest
|
|
|
|
: public MockEventThreadOptsTest,
|
|
|
|
public ::testing::WithParamInterface<std::tuple<ares_evsys_t,int>> {
|
|
|
|
public:
|
|
|
|
MockTCPEventThreadStayOpenTest()
|
|
|
|
: MockEventThreadOptsTest(1, std::get<0>(GetParam()), std::get<1>(GetParam()), true /* tcp */,
|
|
|
|
FillOptions(&opts_),
|
|
|
|
ARES_OPT_FLAGS) {}
|
|
|
|
static struct ares_options* FillOptions(struct ares_options * opts) {
|
|
|
|
memset(opts, 0, sizeof(struct ares_options));
|
|
|
|
opts->flags = ARES_FLAG_STAYOPEN|ARES_FLAG_EDNS;
|
|
|
|
return opts;
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
struct ares_options opts_;
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_P(MockTCPEventThreadStayOpenTest, GetHostByNameParallelLookups) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 100, {2, 3, 4, 5}));
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp));
|
|
|
|
|
|
|
|
// Get notified of new sockets so we can validate how many are created
|
|
|
|
int rc = ARES_SUCCESS;
|
|
|
|
ares_set_socket_callback(channel_, SocketConnectCallback, &rc);
|
|
|
|
sock_cb_count = 0;
|
|
|
|
|
|
|
|
HostResult result[TCPPARALLELLOOKUPS];
|
|
|
|
for (size_t i=0; i<TCPPARALLELLOOKUPS; i++) {
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
Process();
|
|
|
|
|
|
|
|
EXPECT_EQ(1, sock_cb_count);
|
|
|
|
|
|
|
|
for (size_t i=0; i<TCPPARALLELLOOKUPS; i++) {
|
|
|
|
std::stringstream ss;
|
|
|
|
EXPECT_TRUE(result[i].done_);
|
|
|
|
ss << result[i].host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockTCPEventThreadTest, MalformedResponse) {
|
|
|
|
std::vector<byte> one = {0x00};
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReplyData(&server_, one));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ETIMEOUT, result.status_);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockTCPEventThreadTest, FormErrResponse) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A));
|
|
|
|
rsp.set_rcode(FORMERR);
|
|
|
|
EXPECT_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillOnce(SetReply(&server_, &rsp));
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_EFORMERR, result.status_);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockTCPEventThreadTest, ServFailResponse) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A));
|
|
|
|
rsp.set_rcode(SERVFAIL);
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp));
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ESERVFAIL, result.status_);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockTCPEventThreadTest, NotImplResponse) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A));
|
|
|
|
rsp.set_rcode(NOTIMP);
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp));
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ENOTIMP, result.status_);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockTCPEventThreadTest, RefusedResponse) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A));
|
|
|
|
rsp.set_rcode(REFUSED);
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp));
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_EREFUSED, result.status_);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockTCPEventThreadTest, YXDomainResponse) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A));
|
|
|
|
rsp.set_rcode(YXDOMAIN);
|
|
|
|
EXPECT_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillOnce(SetReply(&server_, &rsp));
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ENODATA, result.status_);
|
|
|
|
}
|
|
|
|
|
|
|
|
class MockExtraOptsEventThreadTest
|
|
|
|
: public MockEventThreadOptsTest,
|
|
|
|
public ::testing::WithParamInterface<std::tuple<ares_evsys_t, int, bool> > {
|
|
|
|
public:
|
|
|
|
MockExtraOptsEventThreadTest()
|
|
|
|
: MockEventThreadOptsTest(1, std::get<0>(GetParam()), std::get<1>(GetParam()), std::get<2>(GetParam()),
|
|
|
|
FillOptions(&opts_),
|
|
|
|
ARES_OPT_SOCK_SNDBUF|ARES_OPT_SOCK_RCVBUF) {}
|
|
|
|
static struct ares_options* FillOptions(struct ares_options * opts) {
|
|
|
|
memset(opts, 0, sizeof(struct ares_options));
|
|
|
|
// Set a few options that affect socket communications
|
|
|
|
opts->socket_send_buffer_size = 514;
|
|
|
|
opts->socket_receive_buffer_size = 514;
|
|
|
|
return opts;
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
struct ares_options opts_;
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_P(MockExtraOptsEventThreadTest, SimpleQuery) {
|
|
|
|
ares_set_local_ip4(channel_, 0x7F000001);
|
|
|
|
byte addr6[16] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01};
|
|
|
|
ares_set_local_ip6(channel_, addr6);
|
|
|
|
ares_set_local_dev(channel_, "dummy");
|
|
|
|
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 100, {2, 3, 4, 5}));
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
class MockFlagsEventThreadOptsTest
|
|
|
|
: public MockEventThreadOptsTest,
|
|
|
|
public ::testing::WithParamInterface< std::tuple<ares_evsys_t, int, bool> > {
|
|
|
|
public:
|
|
|
|
MockFlagsEventThreadOptsTest(int flags)
|
|
|
|
: MockEventThreadOptsTest(1, std::get<0>(GetParam()), std::get<1>(GetParam()), std::get<2>(GetParam()),
|
|
|
|
FillOptions(&opts_, flags), ARES_OPT_FLAGS) {}
|
|
|
|
static struct ares_options* FillOptions(struct ares_options * opts, int flags) {
|
|
|
|
memset(opts, 0, sizeof(struct ares_options));
|
|
|
|
opts->flags = flags;
|
|
|
|
return opts;
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
struct ares_options opts_;
|
|
|
|
};
|
|
|
|
|
|
|
|
class MockNoCheckRespEventThreadTest : public MockFlagsEventThreadOptsTest {
|
|
|
|
public:
|
|
|
|
MockNoCheckRespEventThreadTest() : MockFlagsEventThreadOptsTest(ARES_FLAG_NOCHECKRESP) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_P(MockNoCheckRespEventThreadTest, ServFailResponse) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A));
|
|
|
|
rsp.set_rcode(SERVFAIL);
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp));
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ESERVFAIL, result.status_);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockNoCheckRespEventThreadTest, NotImplResponse) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A));
|
|
|
|
rsp.set_rcode(NOTIMP);
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp));
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ENOTIMP, result.status_);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockNoCheckRespEventThreadTest, RefusedResponse) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A));
|
|
|
|
rsp.set_rcode(REFUSED);
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp));
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_EREFUSED, result.status_);
|
|
|
|
}
|
|
|
|
|
|
|
|
class MockEDNSEventThreadTest : public MockFlagsEventThreadOptsTest {
|
|
|
|
public:
|
|
|
|
MockEDNSEventThreadTest() : MockFlagsEventThreadOptsTest(ARES_FLAG_EDNS) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_P(MockEDNSEventThreadTest, RetryWithoutEDNS) {
|
|
|
|
DNSPacket rspfail;
|
|
|
|
rspfail.set_response().set_aa().set_rcode(FORMERR)
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A));
|
|
|
|
DNSPacket rspok;
|
|
|
|
rspok.set_response()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 100, {1, 2, 3, 4}));
|
|
|
|
EXPECT_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillOnce(SetReply(&server_, &rspfail))
|
|
|
|
.WillOnce(SetReply(&server_, &rspok));
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[1.2.3.4]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, SearchDomains) {
|
|
|
|
DNSPacket nofirst;
|
|
|
|
nofirst.set_response().set_aa().set_rcode(NXDOMAIN)
|
|
|
|
.add_question(new DNSQuestion("www.first.com", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www.first.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &nofirst));
|
|
|
|
DNSPacket nosecond;
|
|
|
|
nosecond.set_response().set_aa().set_rcode(NXDOMAIN)
|
|
|
|
.add_question(new DNSQuestion("www.second.org", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www.second.org", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &nosecond));
|
|
|
|
DNSPacket yesthird;
|
|
|
|
yesthird.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.third.gov", T_A))
|
|
|
|
.add_answer(new DNSARR("www.third.gov", 0x0200, {2, 3, 4, 5}));
|
|
|
|
ON_CALL(server_, OnRequest("www.third.gov", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &yesthird));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.third.gov' aliases=[] addrs=[2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Relies on retries so is UDP-only
|
|
|
|
TEST_P(MockUDPEventThreadTest, SearchDomainsWithResentReply) {
|
|
|
|
DNSPacket nofirst;
|
|
|
|
nofirst.set_response().set_aa().set_rcode(NXDOMAIN)
|
|
|
|
.add_question(new DNSQuestion("www.first.com", T_A));
|
|
|
|
EXPECT_CALL(server_, OnRequest("www.first.com", T_A))
|
|
|
|
.WillOnce(SetReply(&server_, &nofirst));
|
|
|
|
DNSPacket nosecond;
|
|
|
|
nosecond.set_response().set_aa().set_rcode(NXDOMAIN)
|
|
|
|
.add_question(new DNSQuestion("www.second.org", T_A));
|
|
|
|
EXPECT_CALL(server_, OnRequest("www.second.org", T_A))
|
|
|
|
.WillOnce(SetReply(&server_, &nosecond));
|
|
|
|
DNSPacket yesthird;
|
|
|
|
yesthird.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.third.gov", T_A))
|
|
|
|
.add_answer(new DNSARR("www.third.gov", 0x0200, {2, 3, 4, 5}));
|
|
|
|
// Before sending the real answer, resend an earlier reply
|
|
|
|
EXPECT_CALL(server_, OnRequest("www.third.gov", T_A))
|
|
|
|
.WillOnce(DoAll(SetReply(&server_, &nofirst),
|
|
|
|
SetReplyQID(&server_, 123)))
|
|
|
|
.WillOnce(DoAll(SetReply(&server_, &yesthird),
|
|
|
|
SetReplyQID(&server_, -1)));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.third.gov' aliases=[] addrs=[2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, SearchDomainsBare) {
|
|
|
|
DNSPacket nofirst;
|
|
|
|
nofirst.set_response().set_aa().set_rcode(NXDOMAIN)
|
|
|
|
.add_question(new DNSQuestion("www.first.com", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www.first.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &nofirst));
|
|
|
|
DNSPacket nosecond;
|
|
|
|
nosecond.set_response().set_aa().set_rcode(NXDOMAIN)
|
|
|
|
.add_question(new DNSQuestion("www.second.org", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www.second.org", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &nosecond));
|
|
|
|
DNSPacket nothird;
|
|
|
|
nothird.set_response().set_aa().set_rcode(NXDOMAIN)
|
|
|
|
.add_question(new DNSQuestion("www.third.gov", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www.third.gov", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, ¬hird));
|
|
|
|
DNSPacket yesbare;
|
|
|
|
yesbare.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www", T_A))
|
|
|
|
.add_answer(new DNSARR("www", 0x0200, {2, 3, 4, 5}));
|
|
|
|
ON_CALL(server_, OnRequest("www", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &yesbare));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www' aliases=[] addrs=[2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, SearchNoDataThenSuccess) {
|
|
|
|
// First two search domains recognize the name but have no A records.
|
|
|
|
DNSPacket nofirst;
|
|
|
|
nofirst.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.first.com", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www.first.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &nofirst));
|
|
|
|
DNSPacket nosecond;
|
|
|
|
nosecond.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.second.org", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www.second.org", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &nosecond));
|
|
|
|
DNSPacket yesthird;
|
|
|
|
yesthird.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.third.gov", T_A))
|
|
|
|
.add_answer(new DNSARR("www.third.gov", 0x0200, {2, 3, 4, 5}));
|
|
|
|
ON_CALL(server_, OnRequest("www.third.gov", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &yesthird));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.third.gov' aliases=[] addrs=[2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, SearchNoDataThenNoDataBare) {
|
|
|
|
// First two search domains recognize the name but have no A records.
|
|
|
|
DNSPacket nofirst;
|
|
|
|
nofirst.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.first.com", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www.first.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &nofirst));
|
|
|
|
DNSPacket nosecond;
|
|
|
|
nosecond.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.second.org", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www.second.org", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &nosecond));
|
|
|
|
DNSPacket nothird;
|
|
|
|
nothird.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.third.gov", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www.third.gov", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, ¬hird));
|
|
|
|
DNSPacket nobare;
|
|
|
|
nobare.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &nobare));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ENODATA, result.status_);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, SearchNoDataThenFail) {
|
|
|
|
// First two search domains recognize the name but have no A records.
|
|
|
|
DNSPacket nofirst;
|
|
|
|
nofirst.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.first.com", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www.first.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &nofirst));
|
|
|
|
DNSPacket nosecond;
|
|
|
|
nosecond.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.second.org", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www.second.org", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &nosecond));
|
|
|
|
DNSPacket nothird;
|
|
|
|
nothird.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.third.gov", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www.third.gov", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, ¬hird));
|
|
|
|
DNSPacket nobare;
|
|
|
|
nobare.set_response().set_aa().set_rcode(NXDOMAIN)
|
|
|
|
.add_question(new DNSQuestion("www", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("www", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &nobare));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ENODATA, result.status_);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, SearchHighNdots) {
|
|
|
|
DNSPacket nobare;
|
|
|
|
nobare.set_response().set_aa().set_rcode(NXDOMAIN)
|
|
|
|
.add_question(new DNSQuestion("a.b.c.w.w.w", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("a.b.c.w.w.w", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &nobare));
|
|
|
|
DNSPacket yesfirst;
|
|
|
|
yesfirst.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("a.b.c.w.w.w.first.com", T_A))
|
|
|
|
.add_answer(new DNSARR("a.b.c.w.w.w.first.com", 0x0200, {2, 3, 4, 5}));
|
|
|
|
ON_CALL(server_, OnRequest("a.b.c.w.w.w.first.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &yesfirst));
|
|
|
|
|
|
|
|
SearchResult result;
|
|
|
|
ares_search(channel_, "a.b.c.w.w.w", C_IN, T_A, SearchCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_SUCCESS, result.status_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << PacketToString(result.data_);
|
|
|
|
EXPECT_EQ("RSP QRY AA NOERROR Q:{'a.b.c.w.w.w.first.com' IN A} "
|
|
|
|
"A:{'a.b.c.w.w.w.first.com' IN A TTL=512 2.3.4.5}",
|
|
|
|
ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, V4WorksV6Timeout) {
|
|
|
|
std::vector<byte> nothing;
|
|
|
|
DNSPacket reply;
|
|
|
|
reply.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 0x0100, {0x01, 0x02, 0x03, 0x04}));
|
|
|
|
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &reply));
|
|
|
|
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_AAAA))
|
|
|
|
.WillByDefault(SetReplyData(&server_, nothing));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_UNSPEC, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(1, result.timeouts_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[1.2.3.4]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, DestroyQuick) {
|
|
|
|
/* We are not looking for any particular result as its possible (but unlikely)
|
|
|
|
* it finished before the destroy completed. We really just want to make sure
|
|
|
|
* cleanup works in this case properly. */
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
ares_destroy(channel_);
|
|
|
|
channel_ = nullptr;
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef CARES_SYMBOL_HIDING
|
|
|
|
// Test case for Issue #662
|
|
|
|
TEST_P(MockEventThreadTest, PartialQueryCancel) {
|
|
|
|
std::vector<byte> nothing;
|
|
|
|
DNSPacket reply;
|
|
|
|
reply.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 0x0100, {0x01, 0x02, 0x03, 0x04}));
|
|
|
|
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &reply));
|
|
|
|
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_AAAA))
|
|
|
|
.WillByDefault(SetReplyData(&server_, nothing));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_UNSPEC, HostCallback, &result);
|
|
|
|
// After 100ms, issues ares_cancel(), this should be enough time for the A
|
|
|
|
// record reply, but before the timeout on the AAAA record.
|
|
|
|
Process(100);
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ECANCELLED, result.status_);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, UnspecifiedFamilyV6) {
|
|
|
|
DNSPacket rsp6;
|
|
|
|
rsp6.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("example.com", T_AAAA))
|
|
|
|
.add_answer(new DNSAaaaRR("example.com", 100,
|
|
|
|
{0x21, 0x21, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0x03}));
|
|
|
|
ON_CALL(server_, OnRequest("example.com", T_AAAA))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp6));
|
|
|
|
|
|
|
|
DNSPacket rsp4;
|
|
|
|
rsp4.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("example.com", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("example.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp4));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "example.com.", AF_UNSPEC, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
// Default to IPv6 when both are available.
|
|
|
|
EXPECT_EQ("{'example.com' aliases=[] addrs=[2121:0000:0000:0000:0000:0000:0000:0303]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, UnspecifiedFamilyV4) {
|
|
|
|
DNSPacket rsp6;
|
|
|
|
rsp6.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("example.com", T_AAAA));
|
|
|
|
ON_CALL(server_, OnRequest("example.com", T_AAAA))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp6));
|
|
|
|
DNSPacket rsp4;
|
|
|
|
rsp4.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("example.com", T_A))
|
|
|
|
.add_answer(new DNSARR("example.com", 100, {2, 3, 4, 5}));
|
|
|
|
ON_CALL(server_, OnRequest("example.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp4));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "example.com.", AF_UNSPEC, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'example.com' aliases=[] addrs=[2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, UnspecifiedFamilyNoData) {
|
|
|
|
DNSPacket rsp6;
|
|
|
|
rsp6.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("example.com", T_AAAA))
|
|
|
|
.add_answer(new DNSCnameRR("example.com", 100, "elsewhere.com"));
|
|
|
|
ON_CALL(server_, OnRequest("example.com", T_AAAA))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp6));
|
|
|
|
DNSPacket rsp4;
|
|
|
|
rsp4.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("example.com", T_A));
|
|
|
|
ON_CALL(server_, OnRequest("example.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp4));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "example.com.", AF_UNSPEC, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'' aliases=[] addrs=[]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, UnspecifiedFamilyCname6A4) {
|
|
|
|
DNSPacket rsp6;
|
|
|
|
rsp6.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("example.com", T_AAAA))
|
|
|
|
.add_answer(new DNSCnameRR("example.com", 100, "elsewhere.com"));
|
|
|
|
ON_CALL(server_, OnRequest("example.com", T_AAAA))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp6));
|
|
|
|
DNSPacket rsp4;
|
|
|
|
rsp4.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("example.com", T_A))
|
|
|
|
.add_answer(new DNSARR("example.com", 100, {1, 2, 3, 4}));
|
|
|
|
ON_CALL(server_, OnRequest("example.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp4));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "example.com.", AF_UNSPEC, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'example.com' aliases=[] addrs=[1.2.3.4]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, ExplicitIP) {
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "1.2.3.4", AF_INET, HostCallback, &result);
|
|
|
|
EXPECT_TRUE(result.done_); // Immediate return
|
|
|
|
EXPECT_EQ(ARES_SUCCESS, result.status_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'1.2.3.4' aliases=[] addrs=[1.2.3.4]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, SortListV4) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("example.com", T_A))
|
|
|
|
.add_answer(new DNSARR("example.com", 100, {22, 23, 24, 25}))
|
|
|
|
.add_answer(new DNSARR("example.com", 100, {12, 13, 14, 15}))
|
|
|
|
.add_answer(new DNSARR("example.com", 100, {2, 3, 4, 5}));
|
|
|
|
ON_CALL(server_, OnRequest("example.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp));
|
|
|
|
|
|
|
|
{
|
|
|
|
EXPECT_EQ(ARES_SUCCESS, ares_set_sortlist(channel_, "12.13.0.0/255.255.0.0 1234::5678"));
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "example.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'example.com' aliases=[] addrs=[12.13.14.15, 22.23.24.25, 2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
{
|
|
|
|
EXPECT_EQ(ARES_SUCCESS, ares_set_sortlist(channel_, "2.3.0.0/16 130.140.150.160/26"));
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "example.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'example.com' aliases=[] addrs=[2.3.4.5, 22.23.24.25, 12.13.14.15]}", ss.str());
|
|
|
|
}
|
|
|
|
struct ares_options options;
|
|
|
|
memset(&options, 0, sizeof(options));
|
|
|
|
int optmask = 0;
|
|
|
|
EXPECT_EQ(ARES_SUCCESS, ares_save_options(channel_, &options, &optmask));
|
|
|
|
EXPECT_TRUE((optmask & ARES_OPT_SORTLIST) == ARES_OPT_SORTLIST);
|
|
|
|
ares_destroy_options(&options);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, SortListV6) {
|
|
|
|
DNSPacket rsp;
|
|
|
|
rsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("example.com", T_AAAA))
|
|
|
|
.add_answer(new DNSAaaaRR("example.com", 100,
|
|
|
|
{0x11, 0x11, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x02}))
|
|
|
|
.add_answer(new DNSAaaaRR("example.com", 100,
|
|
|
|
{0x21, 0x21, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
|
|
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0x03}));
|
|
|
|
ON_CALL(server_, OnRequest("example.com", T_AAAA))
|
|
|
|
.WillByDefault(SetReply(&server_, &rsp));
|
|
|
|
|
|
|
|
{
|
|
|
|
ares_set_sortlist(channel_, "1111::/16 2.3.0.0/255.255.0.0");
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "example.com.", AF_INET6, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'example.com' aliases=[] addrs=[1111:0000:0000:0000:0000:0000:0000:0202, "
|
|
|
|
"2121:0000:0000:0000:0000:0000:0000:0303]}", ss.str());
|
|
|
|
}
|
|
|
|
{
|
|
|
|
ares_set_sortlist(channel_, "2121::/8");
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "example.com.", AF_INET6, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'example.com' aliases=[] addrs=[2121:0000:0000:0000:0000:0000:0000:0303, "
|
|
|
|
"1111:0000:0000:0000:0000:0000:0000:0202]}", ss.str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Relies on retries so is UDP-only
|
|
|
|
TEST_P(MockUDPEventThreadTest, Resend) {
|
|
|
|
std::vector<byte> nothing;
|
|
|
|
DNSPacket reply;
|
|
|
|
reply.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 0x0100, {0x01, 0x02, 0x03, 0x04}));
|
|
|
|
|
|
|
|
EXPECT_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillOnce(SetReplyData(&server_, nothing))
|
|
|
|
.WillOnce(SetReplyData(&server_, nothing))
|
|
|
|
.WillOnce(SetReply(&server_, &reply));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(2, result.timeouts_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[1.2.3.4]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, CancelImmediate) {
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
ares_cancel(channel_);
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ECANCELLED, result.status_);
|
|
|
|
EXPECT_EQ(0, result.timeouts_);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, CancelImmediateGetHostByAddr) {
|
|
|
|
HostResult result;
|
|
|
|
struct in_addr addr;
|
|
|
|
addr.s_addr = htonl(0x08080808);
|
|
|
|
|
|
|
|
ares_gethostbyaddr(channel_, &addr, sizeof(addr), AF_INET, HostCallback, &result);
|
|
|
|
ares_cancel(channel_);
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ECANCELLED, result.status_);
|
|
|
|
EXPECT_EQ(0, result.timeouts_);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Relies on retries so is UDP-only
|
|
|
|
TEST_P(MockUDPEventThreadTest, CancelLater) {
|
|
|
|
std::vector<byte> nothing;
|
|
|
|
|
|
|
|
// On second request, cancel the channel.
|
|
|
|
EXPECT_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillOnce(SetReplyData(&server_, nothing))
|
|
|
|
.WillOnce(CancelChannel(&server_, channel_));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ECANCELLED, result.status_);
|
|
|
|
EXPECT_EQ(0, result.timeouts_);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, DisconnectFirstAttempt) {
|
|
|
|
DNSPacket reply;
|
|
|
|
reply.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 0x0100, {0x01, 0x02, 0x03, 0x04}));
|
|
|
|
|
|
|
|
// On second request, cancel the channel.
|
|
|
|
EXPECT_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillOnce(Disconnect(&server_))
|
|
|
|
.WillOnce(SetReply(&server_, &reply));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.google.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
if (result.done_) {
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[1.2.3.4]}", ss.str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, GetHostByNameCNAMENoData) {
|
|
|
|
DNSPacket response;
|
|
|
|
response.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("cname.first.com", T_A))
|
|
|
|
.add_answer(new DNSCnameRR("cname.first.com", 100, "a.first.com"));
|
|
|
|
ON_CALL(server_, OnRequest("cname.first.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &response));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "cname.first.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_ENODATA, result.status_);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef WIN32
|
|
|
|
TEST_P(MockEventThreadTest, HostAlias) {
|
|
|
|
DNSPacket reply;
|
|
|
|
reply.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.google.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.google.com", 0x0100, {0x01, 0x02, 0x03, 0x04}));
|
|
|
|
ON_CALL(server_, OnRequest("www.google.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &reply));
|
|
|
|
|
|
|
|
TempFile aliases("\n\n# www commentedout\nwww www.google.com\n");
|
|
|
|
EnvValue with_env("HOSTALIASES", aliases.filename());
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.google.com' aliases=[] addrs=[1.2.3.4]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, HostAliasMissing) {
|
|
|
|
DNSPacket yesfirst;
|
|
|
|
yesfirst.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.first.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.first.com", 0x0200, {2, 3, 4, 5}));
|
|
|
|
ON_CALL(server_, OnRequest("www.first.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &yesfirst));
|
|
|
|
|
|
|
|
TempFile aliases("\n\n# www commentedout\nww www.google.com\n");
|
|
|
|
EnvValue with_env("HOSTALIASES", aliases.filename());
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.first.com' aliases=[] addrs=[2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, HostAliasMissingFile) {
|
|
|
|
DNSPacket yesfirst;
|
|
|
|
yesfirst.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.first.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.first.com", 0x0200, {2, 3, 4, 5}));
|
|
|
|
ON_CALL(server_, OnRequest("www.first.com", T_A))
|
|
|
|
.WillByDefault(SetReply(&server_, &yesfirst));
|
|
|
|
|
|
|
|
EnvValue with_env("HOSTALIASES", "bogus.mcfile");
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.first.com' aliases=[] addrs=[2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(MockEventThreadTest, HostAliasUnreadable) {
|
|
|
|
TempFile aliases("www www.google.com\n");
|
|
|
|
EXPECT_EQ(chmod(aliases.filename(), 0), 0);
|
|
|
|
|
|
|
|
/* Perform OS sanity checks. We are observing on Debian after the chmod(fn, 0)
|
|
|
|
* that we are still able to fopen() the file which is unexpected. Skip the
|
|
|
|
* test if we observe this behavior */
|
|
|
|
struct stat st;
|
|
|
|
EXPECT_EQ(stat(aliases.filename(), &st), 0);
|
|
|
|
EXPECT_EQ(st.st_mode & (S_IRWXU|S_IRWXG|S_IRWXO), 0);
|
|
|
|
FILE *fp = fopen(aliases.filename(), "r");
|
|
|
|
if (fp != NULL) {
|
|
|
|
if (verbose) std::cerr << "Skipping Test due to OS incompatibility (open file caching)" << std::endl;
|
|
|
|
fclose(fp);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
EnvValue with_env("HOSTALIASES", aliases.filename());
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(ARES_EFILE, result.status_);
|
|
|
|
chmod(aliases.filename(), 0777);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
class MockMultiServerEventThreadTest
|
|
|
|
: public MockEventThreadOptsTest,
|
|
|
|
public ::testing::WithParamInterface< std::tuple<ares_evsys_t, int, bool> > {
|
|
|
|
public:
|
|
|
|
MockMultiServerEventThreadTest(bool rotate)
|
|
|
|
: MockEventThreadOptsTest(3, std::get<0>(GetParam()), std::get<1>(GetParam()), std::get<2>(GetParam()), nullptr, rotate ? ARES_OPT_ROTATE : ARES_OPT_NOROTATE) {}
|
|
|
|
void CheckExample() {
|
|
|
|
HostResult result;
|
|
|
|
ares_gethostbyname(channel_, "www.example.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << result.host_;
|
|
|
|
EXPECT_EQ("{'www.example.com' aliases=[] addrs=[2.3.4.5]}", ss.str());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class NoRotateMultiMockEventThreadTest : public MockMultiServerEventThreadTest {
|
|
|
|
public:
|
|
|
|
NoRotateMultiMockEventThreadTest() : MockMultiServerEventThreadTest(false) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_P(NoRotateMultiMockEventThreadTest, ThirdServer) {
|
|
|
|
struct ares_options opts;
|
|
|
|
int optmask = 0;
|
|
|
|
memset(&opts, 0, sizeof(opts));
|
|
|
|
EXPECT_EQ(ARES_SUCCESS, ares_save_options(channel_, &opts, &optmask));
|
|
|
|
EXPECT_EQ(ARES_OPT_NOROTATE, (optmask & ARES_OPT_NOROTATE));
|
|
|
|
ares_destroy_options(&opts);
|
|
|
|
|
|
|
|
DNSPacket servfailrsp;
|
|
|
|
servfailrsp.set_response().set_aa().set_rcode(SERVFAIL)
|
|
|
|
.add_question(new DNSQuestion("www.example.com", T_A));
|
|
|
|
DNSPacket notimplrsp;
|
|
|
|
notimplrsp.set_response().set_aa().set_rcode(NOTIMP)
|
|
|
|
.add_question(new DNSQuestion("www.example.com", T_A));
|
|
|
|
DNSPacket okrsp;
|
|
|
|
okrsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.example.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.example.com", 100, {2,3,4,5}));
|
|
|
|
|
|
|
|
EXPECT_CALL(*servers_[0], OnRequest("www.example.com", T_A))
|
|
|
|
.WillOnce(SetReply(servers_[0].get(), &servfailrsp));
|
|
|
|
EXPECT_CALL(*servers_[1], OnRequest("www.example.com", T_A))
|
|
|
|
.WillOnce(SetReply(servers_[1].get(), ¬implrsp));
|
|
|
|
EXPECT_CALL(*servers_[2], OnRequest("www.example.com", T_A))
|
|
|
|
.WillOnce(SetReply(servers_[2].get(), &okrsp));
|
|
|
|
CheckExample();
|
|
|
|
|
|
|
|
// Second time around, still starts from server [2], as [0] and [1] both
|
|
|
|
// recorded failures
|
|
|
|
EXPECT_CALL(*servers_[2], OnRequest("www.example.com", T_A))
|
|
|
|
.WillOnce(SetReply(servers_[2].get(), &servfailrsp));
|
|
|
|
EXPECT_CALL(*servers_[0], OnRequest("www.example.com", T_A))
|
|
|
|
.WillOnce(SetReply(servers_[0].get(), ¬implrsp));
|
|
|
|
EXPECT_CALL(*servers_[1], OnRequest("www.example.com", T_A))
|
|
|
|
.WillOnce(SetReply(servers_[1].get(), &okrsp));
|
|
|
|
CheckExample();
|
|
|
|
|
|
|
|
// Third time around, server order is [1] (f0), [2] (f1), [0] (f2), which
|
|
|
|
// means [1] will get called twice in a row as after the first call
|
|
|
|
// order will be [1] (f1), [2] (f1), [0] (f2) since sort order is
|
|
|
|
// (failure count, index)
|
|
|
|
EXPECT_CALL(*servers_[1], OnRequest("www.example.com", T_A))
|
|
|
|
.WillOnce(SetReply(servers_[1].get(), &servfailrsp))
|
|
|
|
.WillOnce(SetReply(servers_[1].get(), ¬implrsp));
|
|
|
|
EXPECT_CALL(*servers_[2], OnRequest("www.example.com", T_A))
|
|
|
|
.WillOnce(SetReply(servers_[2].get(), ¬implrsp));
|
|
|
|
EXPECT_CALL(*servers_[0], OnRequest("www.example.com", T_A))
|
|
|
|
.WillOnce(SetReply(servers_[0].get(), &okrsp));
|
|
|
|
CheckExample();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_P(NoRotateMultiMockEventThreadTest, ServerNoResponseFailover) {
|
|
|
|
std::vector<byte> nothing;
|
|
|
|
DNSPacket okrsp;
|
|
|
|
okrsp.set_response().set_aa()
|
|
|
|
.add_question(new DNSQuestion("www.example.com", T_A))
|
|
|
|
.add_answer(new DNSARR("www.example.com", 100, {2,3,4,5}));
|
|
|
|
|
|
|
|
/* Server #1 works fine on first attempt, then acts like its offline on
|
|
|
|
* second, then backonline on the third. */
|
|
|
|
EXPECT_CALL(*servers_[0], OnRequest("www.example.com", T_A))
|
|
|
|
.WillOnce(SetReply(servers_[0].get(), &okrsp))
|
|
|
|
.WillOnce(SetReplyData(servers_[0].get(), nothing))
|
|
|
|
.WillOnce(SetReply(servers_[0].get(), &okrsp));
|
|
|
|
|
|
|
|
/* Server #2 always acts like its offline */
|
|
|
|
ON_CALL(*servers_[1], OnRequest("www.example.com", T_A))
|
|
|
|
.WillByDefault(SetReplyData(servers_[1].get(), nothing));
|
|
|
|
|
|
|
|
/* Server #3 works fine on first and second request, then no reply on 3rd */
|
|
|
|
EXPECT_CALL(*servers_[2], OnRequest("www.example.com", T_A))
|
|
|
|
.WillOnce(SetReply(servers_[2].get(), &okrsp))
|
|
|
|
.WillOnce(SetReply(servers_[2].get(), &okrsp))
|
|
|
|
.WillOnce(SetReplyData(servers_[2].get(), nothing));
|
|
|
|
|
|
|
|
HostResult result;
|
|
|
|
|
|
|
|
/* 1. First server returns a response on the first request immediately, normal
|
|
|
|
* operation on channel. */
|
|
|
|
ares_gethostbyname(channel_, "www.example.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(0, result.timeouts_);
|
|
|
|
std::stringstream ss1;
|
|
|
|
ss1 << result.host_;
|
|
|
|
EXPECT_EQ("{'www.example.com' aliases=[] addrs=[2.3.4.5]}", ss1.str());
|
|
|
|
|
|
|
|
/* 2. On the second request, simulate the first and second servers not
|
|
|
|
* returning a response at all, but the 3rd server works, so should have
|
|
|
|
* 2 timeouts. */
|
|
|
|
ares_gethostbyname(channel_, "www.example.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(2, result.timeouts_);
|
|
|
|
std::stringstream ss2;
|
|
|
|
ss2 << result.host_;
|
|
|
|
EXPECT_EQ("{'www.example.com' aliases=[] addrs=[2.3.4.5]}", ss2.str());
|
|
|
|
|
|
|
|
/* 3. On the third request, the active server should be #3, so should respond
|
|
|
|
* immediately with no timeouts */
|
|
|
|
ares_gethostbyname(channel_, "www.example.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(0, result.timeouts_);
|
|
|
|
std::stringstream ss3;
|
|
|
|
ss3 << result.host_;
|
|
|
|
EXPECT_EQ("{'www.example.com' aliases=[] addrs=[2.3.4.5]}", ss3.str());
|
|
|
|
|
|
|
|
/* 4. On the fourth request, the active server should be #3, but will timeout,
|
|
|
|
* and the first server should then respond */
|
|
|
|
ares_gethostbyname(channel_, "www.example.com.", AF_INET, HostCallback, &result);
|
|
|
|
Process();
|
|
|
|
EXPECT_TRUE(result.done_);
|
|
|
|
EXPECT_EQ(1, result.timeouts_);
|
|
|
|
std::stringstream ss4;
|
|
|
|
ss4 << result.host_;
|
|
|
|
EXPECT_EQ("{'www.example.com' aliases=[] addrs=[2.3.4.5]}", ss4.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static const char *evsys_tostr(ares_evsys_t evsys)
|
|
|
|
{
|
|
|
|
switch (evsys) {
|
|
|
|
case ARES_EVSYS_WIN32:
|
|
|
|
return "WIN32";
|
|
|
|
case ARES_EVSYS_EPOLL:
|
|
|
|
return "EPOLL";
|
|
|
|
case ARES_EVSYS_KQUEUE:
|
|
|
|
return "KQUEUE";
|
|
|
|
case ARES_EVSYS_POLL:
|
|
|
|
return "POLL";
|
|
|
|
case ARES_EVSYS_SELECT:
|
|
|
|
return "SELECT";
|
|
|
|
case ARES_EVSYS_DEFAULT:
|
|
|
|
return "DEFAULT";
|
|
|
|
}
|
|
|
|
return "UNKNOWN";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static std::string PrintEvsysFamilyMode(const testing::TestParamInfo<std::tuple<ares_evsys_t, int, bool>> &info)
|
|
|
|
{
|
|
|
|
std::string name;
|
|
|
|
|
|
|
|
name += evsys_tostr(std::get<0>(info.param));
|
|
|
|
name += "_";
|
|
|
|
name += af_tostr(std::get<1>(info.param));
|
|
|
|
name += "_";
|
|
|
|
name += mode_tostr(std::get<2>(info.param));
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::string PrintEvsysFamily(const testing::TestParamInfo<std::tuple<ares_evsys_t, int>> &info)
|
|
|
|
{
|
|
|
|
std::string name;
|
|
|
|
|
|
|
|
name += evsys_tostr(std::get<0>(info.param));
|
|
|
|
name += "_";
|
|
|
|
name += af_tostr(std::get<1>(info.param));
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_SUITE_P(AddressFamilies, MockEventThreadTest, ::testing::ValuesIn(ares::test::evsys_families_modes), ares::test::PrintEvsysFamilyMode);
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_SUITE_P(AddressFamilies, MockUDPEventThreadTest, ::testing::ValuesIn(ares::test::evsys_families), ares::test::PrintEvsysFamily);
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_SUITE_P(AddressFamilies, MockUDPEventThreadMaxQueriesTest, ::testing::ValuesIn(ares::test::evsys_families), ares::test::PrintEvsysFamily);
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_SUITE_P(AddressFamilies, CacheQueriesEventThreadTest, ::testing::ValuesIn(ares::test::evsys_families), ares::test::PrintEvsysFamily);
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_SUITE_P(AddressFamilies, MockTCPEventThreadTest, ::testing::ValuesIn(ares::test::evsys_families), ares::test::PrintEvsysFamily);
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_SUITE_P(AddressFamilies, MockTCPEventThreadStayOpenTest, ::testing::ValuesIn(ares::test::evsys_families), ares::test::PrintEvsysFamily);
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_SUITE_P(AddressFamilies, MockExtraOptsEventThreadTest, ::testing::ValuesIn(ares::test::evsys_families_modes), ares::test::PrintEvsysFamilyMode);
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_SUITE_P(AddressFamilies, MockNoCheckRespEventThreadTest, ::testing::ValuesIn(ares::test::evsys_families_modes), ares::test::PrintEvsysFamilyMode);
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_SUITE_P(AddressFamilies, MockEDNSEventThreadTest, ::testing::ValuesIn(ares::test::evsys_families_modes), ares::test::PrintEvsysFamilyMode);
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_SUITE_P(TransportModes, NoRotateMultiMockEventThreadTest, ::testing::ValuesIn(ares::test::evsys_families_modes), ares::test::PrintEvsysFamilyMode);
|
|
|
|
|
|
|
|
} // namespace test
|
|
|
|
} // namespace ares
|
|
|
|
|
|
|
|
#endif /* CARES_THREADS */
|