test-ip-rule.cc 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306
  1. /*************************************************************************
  2. *
  3. * Copyright (C) 2018-2025 Ruilin Peng (Nick) <[email protected]>.
  4. *
  5. * smartdns is free software: you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation, either version 3 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * smartdns is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  17. */
  18. #include "client.h"
  19. #include "smartdns/dns.h"
  20. #include "include/utils.h"
  21. #include "server.h"
  22. #include "gtest/gtest.h"
  23. class IPRule : public ::testing::Test
  24. {
  25. protected:
  26. virtual void SetUp() {}
  27. virtual void TearDown() {}
  28. };
  29. TEST_F(IPRule, white_list)
  30. {
  31. smartdns::MockServer server_upstream;
  32. smartdns::MockServer server_upstream2;
  33. smartdns::Server server;
  34. server_upstream.Start("udp://0.0.0.0:61053", [](struct smartdns::ServerRequestContext *request) {
  35. if (request->qtype != DNS_T_A) {
  36. return smartdns::SERVER_REQUEST_SOA;
  37. }
  38. smartdns::MockServer::AddIP(request, request->domain.c_str(), "1.2.3.4", 611);
  39. return smartdns::SERVER_REQUEST_OK;
  40. });
  41. server_upstream2.Start("udp://0.0.0.0:62053", [](struct smartdns::ServerRequestContext *request) {
  42. if (request->qtype != DNS_T_A) {
  43. return smartdns::SERVER_REQUEST_SOA;
  44. }
  45. smartdns::MockServer::AddIP(request, request->domain.c_str(), "4.5.6.7", 611);
  46. return smartdns::SERVER_REQUEST_OK;
  47. });
  48. /* this ip will be discard, but is reachable */
  49. server.MockPing(PING_TYPE_ICMP, "1.2.3.4", 60, 10);
  50. server.Start(R"""(bind [::]:60053
  51. server udp://127.0.0.1:61053 -whitelist-ip
  52. server udp://127.0.0.1:62053 -whitelist-ip
  53. whitelist-ip 4.5.6.7/24
  54. )""");
  55. smartdns::Client client;
  56. ASSERT_TRUE(client.Query("a.com", 60053));
  57. std::cout << client.GetResult() << std::endl;
  58. ASSERT_EQ(client.GetAnswerNum(), 1);
  59. EXPECT_EQ(client.GetStatus(), "NOERROR");
  60. EXPECT_EQ(client.GetAnswer()[0].GetName(), "a.com");
  61. EXPECT_EQ(client.GetAnswer()[0].GetData(), "4.5.6.7");
  62. }
  63. TEST_F(IPRule, white_list_not_in)
  64. {
  65. smartdns::MockServer server_upstream;
  66. smartdns::MockServer server_upstream2;
  67. smartdns::Server server;
  68. server_upstream.Start("udp://0.0.0.0:61053", [](struct smartdns::ServerRequestContext *request) {
  69. if (request->qtype != DNS_T_A) {
  70. return smartdns::SERVER_REQUEST_SOA;
  71. }
  72. smartdns::MockServer::AddIP(request, request->domain.c_str(), "1.2.3.4", 611);
  73. return smartdns::SERVER_REQUEST_OK;
  74. });
  75. server_upstream2.Start("udp://0.0.0.0:62053", [](struct smartdns::ServerRequestContext *request) {
  76. if (request->qtype != DNS_T_A) {
  77. return smartdns::SERVER_REQUEST_SOA;
  78. }
  79. smartdns::MockServer::AddIP(request, request->domain.c_str(), "9.10.11.12", 611);
  80. return smartdns::SERVER_REQUEST_OK;
  81. });
  82. /* this ip will be discard, but is reachable */
  83. server.MockPing(PING_TYPE_ICMP, "1.2.3.4", 60, 10);
  84. server.Start(R"""(bind [::]:60053
  85. server udp://127.0.0.1:61053 -whitelist-ip
  86. server udp://127.0.0.1:62053 -whitelist-ip
  87. whitelist-ip 4.5.6.7/24
  88. )""");
  89. smartdns::Client client;
  90. ASSERT_TRUE(client.Query("a.com", 60053));
  91. std::cout << client.GetResult() << std::endl;
  92. ASSERT_EQ(client.GetAnswerNum(), 0);
  93. EXPECT_EQ(client.GetStatus(), "SERVFAIL");
  94. }
  95. TEST_F(IPRule, black_list)
  96. {
  97. smartdns::MockServer server_upstream;
  98. smartdns::MockServer server_upstream2;
  99. smartdns::Server server;
  100. server_upstream.Start("udp://0.0.0.0:61053", [](struct smartdns::ServerRequestContext *request) {
  101. if (request->qtype != DNS_T_A) {
  102. return smartdns::SERVER_REQUEST_SOA;
  103. }
  104. usleep(800000);
  105. smartdns::MockServer::AddIP(request, request->domain.c_str(), "1.2.3.4", 611);
  106. return smartdns::SERVER_REQUEST_OK;
  107. });
  108. server_upstream2.Start("udp://0.0.0.0:62053", [](struct smartdns::ServerRequestContext *request) {
  109. if (request->qtype != DNS_T_A) {
  110. return smartdns::SERVER_REQUEST_SOA;
  111. }
  112. smartdns::MockServer::AddIP(request, request->domain.c_str(), "4.5.6.7", 611);
  113. return smartdns::SERVER_REQUEST_OK;
  114. });
  115. /* this ip will be discard, but is reachable */
  116. server.MockPing(PING_TYPE_ICMP, "4.5.6.7", 60, 10);
  117. server.Start(R"""(bind [::]:60053
  118. server udp://127.0.0.1:61053 -blacklist-ip
  119. server udp://127.0.0.1:62053 -blacklist-ip
  120. blacklist-ip 4.5.6.7/24
  121. )""");
  122. smartdns::Client client;
  123. ASSERT_TRUE(client.Query("a.com", 60053));
  124. std::cout << client.GetResult() << std::endl;
  125. ASSERT_EQ(client.GetAnswerNum(), 1);
  126. EXPECT_EQ(client.GetStatus(), "NOERROR");
  127. EXPECT_EQ(client.GetAnswer()[0].GetName(), "a.com");
  128. EXPECT_EQ(client.GetAnswer()[0].GetData(), "1.2.3.4");
  129. }
  130. TEST_F(IPRule, ignore_ip)
  131. {
  132. smartdns::MockServer server_upstream;
  133. smartdns::MockServer server_upstream2;
  134. smartdns::Server server;
  135. server_upstream.Start("udp://0.0.0.0:61053", [](struct smartdns::ServerRequestContext *request) {
  136. if (request->qtype != DNS_T_A) {
  137. return smartdns::SERVER_REQUEST_SOA;
  138. }
  139. smartdns::MockServer::AddIP(request, request->domain.c_str(), "1.2.3.4", 611);
  140. smartdns::MockServer::AddIP(request, request->domain.c_str(), "4.5.6.7", 611);
  141. smartdns::MockServer::AddIP(request, request->domain.c_str(), "7.8.9.10", 611);
  142. return smartdns::SERVER_REQUEST_OK;
  143. });
  144. /* this ip will be discard, but is reachable */
  145. server.MockPing(PING_TYPE_ICMP, "1.2.3.4", 60, 10);
  146. server.MockPing(PING_TYPE_ICMP, "4.5.6.7", 60, 90);
  147. server.MockPing(PING_TYPE_ICMP, "7.8.9.10", 60, 40);
  148. server.Start(R"""(bind [::]:60053
  149. server udp://127.0.0.1:61053 -blacklist-ip
  150. ignore-ip 1.2.3.0/24
  151. )""");
  152. smartdns::Client client;
  153. ASSERT_TRUE(client.Query("a.com", 60053));
  154. std::cout << client.GetResult() << std::endl;
  155. ASSERT_EQ(client.GetAnswerNum(), 1);
  156. EXPECT_EQ(client.GetStatus(), "NOERROR");
  157. EXPECT_EQ(client.GetAnswer()[0].GetName(), "a.com");
  158. EXPECT_EQ(client.GetAnswer()[0].GetData(), "7.8.9.10");
  159. }
  160. TEST_F(IPRule, ignore_ip_set)
  161. {
  162. smartdns::MockServer server_upstream;
  163. smartdns::MockServer server_upstream2;
  164. smartdns::Server server;
  165. std::string file = "/tmp/smartdns_test_ip_set.list" + smartdns::GenerateRandomString(5);
  166. std::ofstream ofs(file);
  167. ASSERT_TRUE(ofs.is_open());
  168. Defer
  169. {
  170. ofs.close();
  171. unlink(file.c_str());
  172. };
  173. server_upstream.Start("udp://0.0.0.0:61053", [](struct smartdns::ServerRequestContext *request) {
  174. if (request->qtype != DNS_T_A) {
  175. return smartdns::SERVER_REQUEST_SOA;
  176. }
  177. smartdns::MockServer::AddIP(request, request->domain.c_str(), "1.2.3.4", 611);
  178. smartdns::MockServer::AddIP(request, request->domain.c_str(), "4.5.6.7", 611);
  179. smartdns::MockServer::AddIP(request, request->domain.c_str(), "7.8.9.10", 611);
  180. return smartdns::SERVER_REQUEST_OK;
  181. });
  182. /* this ip will be discard, but is reachable */
  183. server.MockPing(PING_TYPE_ICMP, "1.2.3.4", 60, 10);
  184. server.MockPing(PING_TYPE_ICMP, "4.5.6.7", 60, 90);
  185. server.MockPing(PING_TYPE_ICMP, "7.8.9.10", 60, 40);
  186. std::string ipset_list = R"""(
  187. 1.2.3.0/24
  188. 4.5.6.0/24
  189. )""";
  190. ofs.write(ipset_list.c_str(), ipset_list.length());
  191. ofs.flush();
  192. server.Start(R"""(bind [::]:60053
  193. server udp://127.0.0.1:61053 -blacklist-ip
  194. ip-set -name ip-list -file )""" +
  195. file + R"""(
  196. ignore-ip ip-set:ip-list
  197. speed-check-mode none
  198. )""");
  199. smartdns::Client client;
  200. ASSERT_TRUE(client.Query("a.com", 60053));
  201. std::cout << client.GetResult() << std::endl;
  202. ASSERT_EQ(client.GetAnswerNum(), 1);
  203. EXPECT_EQ(client.GetStatus(), "NOERROR");
  204. EXPECT_EQ(client.GetAnswer()[0].GetName(), "a.com");
  205. EXPECT_EQ(client.GetAnswer()[0].GetData(), "7.8.9.10");
  206. }
  207. TEST_F(IPRule, ip_alias_ip_set)
  208. {
  209. smartdns::MockServer server_upstream;
  210. smartdns::MockServer server_upstream2;
  211. smartdns::Server server;
  212. std::string file = "/tmp/smartdns_test_ip_set.list" + smartdns::GenerateRandomString(5);
  213. std::string file_ip = "/tmp/smartdns_test_ip_set_ip.list" + smartdns::GenerateRandomString(5);
  214. std::ofstream ofs(file);
  215. std::ofstream ofs_ip(file_ip);
  216. ASSERT_TRUE(ofs.is_open());
  217. ASSERT_TRUE(ofs_ip.is_open());
  218. Defer
  219. {
  220. ofs.close();
  221. unlink(file.c_str());
  222. ofs_ip.close();
  223. unlink(file_ip.c_str());
  224. };
  225. server_upstream.Start("udp://0.0.0.0:61053", [](struct smartdns::ServerRequestContext *request) {
  226. if (request->qtype != DNS_T_A) {
  227. return smartdns::SERVER_REQUEST_SOA;
  228. }
  229. smartdns::MockServer::AddIP(request, request->domain.c_str(), "1.2.3.4", 611);
  230. smartdns::MockServer::AddIP(request, request->domain.c_str(), "4.5.6.7", 611);
  231. smartdns::MockServer::AddIP(request, request->domain.c_str(), "7.8.9.10", 611);
  232. return smartdns::SERVER_REQUEST_OK;
  233. });
  234. server.MockPing(PING_TYPE_ICMP, "1.2.3.4", 60, 10);
  235. server.MockPing(PING_TYPE_ICMP, "4.5.6.7", 60, 90);
  236. server.MockPing(PING_TYPE_ICMP, "7.8.9.10", 60, 40);
  237. std::string ipset_list = R"""(
  238. 1.2.3.0/24
  239. 4.5.6.0/24
  240. 7.8.9.0/24
  241. )""";
  242. ofs.write(ipset_list.c_str(), ipset_list.length());
  243. ofs.flush();
  244. std::string ipset_list_ip = R"""(
  245. 1.1.1.1
  246. )""";
  247. ofs_ip.write(ipset_list_ip.c_str(), ipset_list_ip.length());
  248. ofs_ip.flush();
  249. server.Start(R"""(bind [::]:60053
  250. server udp://127.0.0.1:61053 -blacklist-ip
  251. ip-set -name ip-list -file )""" +
  252. file + R"""(
  253. ip-set -name ip-list-ip -file )""" +
  254. file_ip + R"""(
  255. ip-alias ip-set:ip-list ip-set:ip-list-ip
  256. speed-check-mode none
  257. )""");
  258. smartdns::Client client;
  259. ASSERT_TRUE(client.Query("a.com", 60053));
  260. std::cout << client.GetResult() << std::endl;
  261. ASSERT_EQ(client.GetAnswerNum(), 1);
  262. EXPECT_EQ(client.GetStatus(), "NOERROR");
  263. EXPECT_EQ(client.GetAnswer()[0].GetName(), "a.com");
  264. EXPECT_EQ(client.GetAnswer()[0].GetData(), "1.1.1.1");
  265. }