/************************************************************************* * * Copyright (C) 2018-2023 Ruilin Peng (Nick) . * * smartdns is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * smartdns is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include "client.h" #include "include/utils.h" #include "server.h" #include "gtest/gtest.h" #include #include #include #include #include TEST(Bind, tls) { Defer { unlink("/tmp/smartdns-cert.pem"); unlink("/tmp/smartdns-key.pem"); }; smartdns::Server server_wrap; smartdns::Server server; server.Start(R"""(bind [::]:61053 server-tls 127.0.0.1:60053 -no-check-certificate log-num 0 log-console yes log-level debug cache-persist no)"""); server_wrap.Start(R"""(bind-tls [::]:60053 address /example.com/1.2.3.4 log-num 0 log-console yes log-level debug cache-persist no)"""); smartdns::Client client; ASSERT_TRUE(client.Query("example.com", 61053)); ASSERT_EQ(client.GetAnswerNum(), 1); EXPECT_EQ(client.GetStatus(), "NOERROR"); EXPECT_EQ(client.GetAnswer()[0].GetData(), "1.2.3.4"); } TEST(Bind, udp_tcp) { smartdns::MockServer server_upstream; smartdns::MockServer server_upstream2; smartdns::Server server; server_upstream.Start("udp://0.0.0.0:61053", [](struct smartdns::ServerRequestContext *request) { unsigned char addr[4] = {1, 2, 3, 4}; dns_add_A(request->response_packet, DNS_RRS_AN, request->domain.c_str(), 611, addr); request->response_packet->head.rcode = DNS_RC_NOERROR; return smartdns::SERVER_REQUEST_OK; }); server.Start(R"""( bind [::]:60053 bind-tcp [::]:60053 server 127.0.0.1:61053 log-num 0 log-console yes log-level debug cache-persist no)"""); smartdns::Client client; ASSERT_TRUE(client.Query("a.com +tcp", 60053)); std::cout << client.GetResult() << std::endl; ASSERT_EQ(client.GetAnswerNum(), 1); EXPECT_EQ(client.GetStatus(), "NOERROR"); EXPECT_EQ(client.GetAnswer()[0].GetTTL(), 3); EXPECT_EQ(client.GetAnswer()[0].GetData(), "1.2.3.4"); ASSERT_TRUE(client.Query("a.com", 60053)); std::cout << client.GetResult() << std::endl; ASSERT_EQ(client.GetAnswerNum(), 1); EXPECT_EQ(client.GetStatus(), "NOERROR"); EXPECT_GE(client.GetAnswer()[0].GetTTL(), 609); EXPECT_EQ(client.GetAnswer()[0].GetData(), "1.2.3.4"); } TEST(Bind, self) { smartdns::MockServer server_upstream; smartdns::Server server; server_upstream.Start("udp://0.0.0.0:62053", [](struct smartdns::ServerRequestContext *request) { if (request->qtype == DNS_T_A) { smartdns::MockServer::AddIP(request, request->domain.c_str(), "1.2.3.4"); return smartdns::SERVER_REQUEST_OK; } return smartdns::SERVER_REQUEST_SOA; }); server.Start(R"""( bind [::]:60053 -group self server 127.0.0.1:61053 -group self bind [::]:61053 -group upstream server 127.0.0.1:62053 -group upstream log-num 0 log-console yes log-level info cache-persist no)"""); smartdns::Client client; ASSERT_TRUE(client.Query("a.com", 60053)); std::cout << client.GetResult() << std::endl; ASSERT_EQ(client.GetAnswerNum(), 1); EXPECT_EQ(client.GetStatus(), "NOERROR"); EXPECT_LT(client.GetQueryTime(), 100); EXPECT_EQ(client.GetAnswer()[0].GetTTL(), 3); EXPECT_EQ(client.GetAnswer()[0].GetData(), "1.2.3.4"); } TEST(Bind, nocache) { smartdns::MockServer server_upstream; smartdns::MockServer server_upstream2; smartdns::Server server; server_upstream.Start("udp://0.0.0.0:61053", [](struct smartdns::ServerRequestContext *request) { unsigned char addr[4] = {1, 2, 3, 4}; usleep(15 * 1000); dns_add_A(request->response_packet, DNS_RRS_AN, request->domain.c_str(), 611, addr); request->response_packet->head.rcode = DNS_RC_NOERROR; return smartdns::SERVER_REQUEST_OK; }); server.Start(R"""( bind [::]:60053 --no-cache bind-tcp [::]:60053 server 127.0.0.1:61053 log-num 0 log-console yes log-level debug cache-persist no)"""); smartdns::Client client; ASSERT_TRUE(client.Query("a.com", 60053)); std::cout << client.GetResult() << std::endl; ASSERT_EQ(client.GetAnswerNum(), 1); EXPECT_EQ(client.GetStatus(), "NOERROR"); EXPECT_EQ(client.GetAnswer()[0].GetName(), "a.com"); EXPECT_EQ(client.GetAnswer()[0].GetTTL(), 3); EXPECT_EQ(client.GetAnswer()[0].GetData(), "1.2.3.4"); ASSERT_TRUE(client.Query("a.com", 60053)); EXPECT_GT(client.GetQueryTime(), 10); std::cout << client.GetResult() << std::endl; ASSERT_EQ(client.GetAnswerNum(), 1); EXPECT_EQ(client.GetStatus(), "NOERROR"); EXPECT_EQ(client.GetAnswer()[0].GetName(), "a.com"); EXPECT_EQ(client.GetAnswer()[0].GetTTL(), 3); EXPECT_EQ(client.GetAnswer()[0].GetData(), "1.2.3.4"); } TEST(Bind, device) { smartdns::MockServer server_upstream; smartdns::Server server; server_upstream.Start("udp://0.0.0.0:62053", [](struct smartdns::ServerRequestContext *request) { if (request->qtype == DNS_T_A) { smartdns::MockServer::AddIP(request, request->domain.c_str(), "1.2.3.4"); return smartdns::SERVER_REQUEST_OK; } return smartdns::SERVER_REQUEST_SOA; }); server.Start(R"""( bind [::]:60053@lo server 127.0.0.1:62053 log-num 0 log-console yes log-level info cache-persist no)"""); smartdns::Client client; ASSERT_TRUE(client.Query("a.com", 60053)); std::cout << client.GetResult() << std::endl; ASSERT_EQ(client.GetAnswerNum(), 1); EXPECT_EQ(client.GetStatus(), "NOERROR"); EXPECT_LT(client.GetQueryTime(), 100); EXPECT_EQ(client.GetAnswer()[0].GetTTL(), 3); EXPECT_EQ(client.GetAnswer()[0].GetData(), "1.2.3.4"); } TEST(Bind, malformed_packet) { smartdns::MockServer server_upstream; smartdns::Server server; server_upstream.Start("udp://0.0.0.0:62053", [](struct smartdns::ServerRequestContext *request) { if (request->qtype == DNS_T_A) { smartdns::MockServer::AddIP(request, request->domain.c_str(), "1.2.3.4"); return smartdns::SERVER_REQUEST_OK; } return smartdns::SERVER_REQUEST_SOA; }); server.Start(R"""( bind [::]:60053@lo server 127.0.0.1:62053 log-num 0 log-console yes log-level info cache-persist no)"""); int sockfd = socket(AF_INET, SOCK_DGRAM, 0); ASSERT_NE(sockfd, -1); Defer { close(sockfd); }; struct sockaddr_in server_addr; server_addr.sin_family = AF_INET; server_addr.sin_port = htons(60053); server_addr.sin_addr.s_addr = inet_addr("127.0.0.1"); for (int i = 0; i < 100000; i++) { char buf[4096]; int len = random() % 4096; if (len <= 0) { len = 1; } RAND_bytes((unsigned char *)buf, len); sendto(sockfd, buf, len, 0, (struct sockaddr *)&server_addr, sizeof(server_addr)); } smartdns::Client client; ASSERT_TRUE(client.Query("a.com", 60053)); std::cout << client.GetResult() << std::endl; ASSERT_EQ(client.GetAnswerNum(), 1); EXPECT_EQ(client.GetStatus(), "NOERROR"); EXPECT_LT(client.GetQueryTime(), 100); EXPECT_EQ(client.GetAnswer()[0].GetTTL(), 3); EXPECT_EQ(client.GetAnswer()[0].GetData(), "1.2.3.4"); }