test-subnet.cc 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447
  1. /*************************************************************************
  2. *
  3. * Copyright (C) 2018-2023 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 "dns.h"
  20. #include "include/utils.h"
  21. #include "server.h"
  22. #include "util.h"
  23. #include "gtest/gtest.h"
  24. #include <fstream>
  25. class SubNet : public ::testing::Test
  26. {
  27. protected:
  28. virtual void SetUp() {}
  29. virtual void TearDown() {}
  30. };
  31. TEST_F(SubNet, pass_subnet)
  32. {
  33. smartdns::MockServer server_upstream;
  34. smartdns::Server server;
  35. server_upstream.Start("udp://0.0.0.0:61053", [&](struct smartdns::ServerRequestContext *request) {
  36. if (request->qtype != DNS_T_A) {
  37. return smartdns::SERVER_REQUEST_SOA;
  38. }
  39. struct dns_opt_ecs ecs;
  40. struct dns_rrs *rrs = NULL;
  41. int rr_count = 0;
  42. int i = 0;
  43. int ret = 0;
  44. int has_ecs = 0;
  45. rr_count = 0;
  46. rrs = dns_get_rrs_start(request->packet, DNS_RRS_OPT, &rr_count);
  47. if (rr_count <= 0) {
  48. return smartdns::SERVER_REQUEST_ERROR;
  49. }
  50. for (i = 0; i < rr_count && rrs; i++, rrs = dns_get_rrs_next(request->packet, rrs)) {
  51. memset(&ecs, 0, sizeof(ecs));
  52. ret = dns_get_OPT_ECS(rrs, NULL, NULL, &ecs);
  53. if (ret != 0) {
  54. continue;
  55. }
  56. has_ecs = 1;
  57. break;
  58. }
  59. if (has_ecs == 0) {
  60. return smartdns::SERVER_REQUEST_ERROR;
  61. }
  62. if (ecs.family != 1) {
  63. return smartdns::SERVER_REQUEST_ERROR;
  64. }
  65. if (memcmp(ecs.addr, "\x08\x08\x08\x00", 4) != 0) {
  66. return smartdns::SERVER_REQUEST_ERROR;
  67. }
  68. if (ecs.source_prefix != 24) {
  69. return smartdns::SERVER_REQUEST_ERROR;
  70. }
  71. smartdns::MockServer::AddIP(request, request->domain.c_str(), "1.2.3.4");
  72. return smartdns::SERVER_REQUEST_OK;
  73. });
  74. server.Start(R"""(bind [::]:60053
  75. server 127.0.0.1:61053
  76. log-num 0
  77. log-console yes
  78. dualstack-ip-selection no
  79. log-level debug
  80. cache-persist no)""");
  81. smartdns::Client client;
  82. ASSERT_TRUE(client.Query("a.com A +subnet=8.8.8.8/24", 60053));
  83. std::cout << client.GetResult() << std::endl;
  84. ASSERT_EQ(client.GetAnswerNum(), 1);
  85. EXPECT_EQ(client.GetStatus(), "NOERROR");
  86. EXPECT_EQ(client.GetAnswer()[0].GetName(), "a.com");
  87. EXPECT_EQ(client.GetAnswer()[0].GetTTL(), 3);
  88. EXPECT_EQ(client.GetAnswer()[0].GetType(), "A");
  89. EXPECT_EQ(client.GetAnswer()[0].GetData(), "1.2.3.4");
  90. }
  91. TEST_F(SubNet, conf)
  92. {
  93. smartdns::MockServer server_upstream;
  94. smartdns::Server server;
  95. server_upstream.Start("udp://0.0.0.0:61053", [&](struct smartdns::ServerRequestContext *request) {
  96. if (request->qtype != DNS_T_A) {
  97. return smartdns::SERVER_REQUEST_SOA;
  98. }
  99. struct dns_opt_ecs ecs;
  100. struct dns_rrs *rrs = NULL;
  101. int rr_count = 0;
  102. int i = 0;
  103. int ret = 0;
  104. int has_ecs = 0;
  105. rr_count = 0;
  106. rrs = dns_get_rrs_start(request->packet, DNS_RRS_OPT, &rr_count);
  107. if (rr_count <= 0) {
  108. return smartdns::SERVER_REQUEST_ERROR;
  109. }
  110. for (i = 0; i < rr_count && rrs; i++, rrs = dns_get_rrs_next(request->packet, rrs)) {
  111. memset(&ecs, 0, sizeof(ecs));
  112. ret = dns_get_OPT_ECS(rrs, NULL, NULL, &ecs);
  113. if (ret != 0) {
  114. continue;
  115. }
  116. has_ecs = 1;
  117. break;
  118. }
  119. if (has_ecs == 0) {
  120. return smartdns::SERVER_REQUEST_ERROR;
  121. }
  122. if (ecs.family != DNS_OPT_ECS_FAMILY_IPV4) {
  123. return smartdns::SERVER_REQUEST_ERROR;
  124. }
  125. if (memcmp(ecs.addr, "\x08\x08\x08\x00", 4) != 0) {
  126. return smartdns::SERVER_REQUEST_ERROR;
  127. }
  128. if (ecs.source_prefix != 24) {
  129. return smartdns::SERVER_REQUEST_ERROR;
  130. }
  131. smartdns::MockServer::AddIP(request, request->domain.c_str(), "1.2.3.4");
  132. return smartdns::SERVER_REQUEST_OK;
  133. });
  134. server.Start(R"""(bind [::]:60053
  135. server 127.0.0.1:61053
  136. log-num 0
  137. log-console yes
  138. dualstack-ip-selection no
  139. edns-client-subnet 8.8.8.8/24
  140. log-level debug
  141. cache-persist no)""");
  142. smartdns::Client client;
  143. ASSERT_TRUE(client.Query("a.com A", 60053));
  144. std::cout << client.GetResult() << std::endl;
  145. ASSERT_EQ(client.GetAnswerNum(), 1);
  146. EXPECT_EQ(client.GetStatus(), "NOERROR");
  147. EXPECT_EQ(client.GetAnswer()[0].GetName(), "a.com");
  148. EXPECT_EQ(client.GetAnswer()[0].GetTTL(), 3);
  149. EXPECT_EQ(client.GetAnswer()[0].GetType(), "A");
  150. EXPECT_EQ(client.GetAnswer()[0].GetData(), "1.2.3.4");
  151. }
  152. TEST_F(SubNet, conf_v6)
  153. {
  154. smartdns::MockServer server_upstream;
  155. smartdns::Server server;
  156. server_upstream.Start("udp://0.0.0.0:61053", [&](struct smartdns::ServerRequestContext *request) {
  157. if (request->qtype != DNS_T_AAAA) {
  158. return smartdns::SERVER_REQUEST_SOA;
  159. }
  160. struct dns_opt_ecs ecs;
  161. struct dns_rrs *rrs = NULL;
  162. int rr_count = 0;
  163. int i = 0;
  164. int ret = 0;
  165. int has_ecs = 0;
  166. rr_count = 0;
  167. rrs = dns_get_rrs_start(request->packet, DNS_RRS_OPT, &rr_count);
  168. if (rr_count <= 0) {
  169. return smartdns::SERVER_REQUEST_ERROR;
  170. }
  171. for (i = 0; i < rr_count && rrs; i++, rrs = dns_get_rrs_next(request->packet, rrs)) {
  172. memset(&ecs, 0, sizeof(ecs));
  173. ret = dns_get_OPT_ECS(rrs, NULL, NULL, &ecs);
  174. if (ret != 0) {
  175. continue;
  176. }
  177. has_ecs = 1;
  178. break;
  179. }
  180. if (has_ecs == 0) {
  181. return smartdns::SERVER_REQUEST_ERROR;
  182. }
  183. if (ecs.family != DNS_OPT_ECS_FAMILY_IPV6) {
  184. return smartdns::SERVER_REQUEST_ERROR;
  185. }
  186. if (memcmp(ecs.addr, "\xff\xff\xff\xff\xff\xff\xff\xff\x00\x00\x00\x00\x00\x00\x00\x00", 16) != 0) {
  187. return smartdns::SERVER_REQUEST_ERROR;
  188. }
  189. if (ecs.source_prefix != 64) {
  190. return smartdns::SERVER_REQUEST_ERROR;
  191. }
  192. smartdns::MockServer::AddIP(request, request->domain.c_str(), "2001:db8::1");
  193. return smartdns::SERVER_REQUEST_OK;
  194. });
  195. server.MockPing(PING_TYPE_ICMP, "2001:db8::1", 60, 70);
  196. server.Start(R"""(bind [::]:60053
  197. server 127.0.0.1:61053
  198. log-num 0
  199. log-console yes
  200. dualstack-ip-selection no
  201. edns-client-subnet ffff:ffff:ffff:ffff:ffff::/64
  202. log-level debug
  203. cache-persist no)""");
  204. smartdns::Client client;
  205. ASSERT_TRUE(client.Query("a.com AAAA", 60053));
  206. std::cout << client.GetResult() << std::endl;
  207. ASSERT_EQ(client.GetAnswerNum(), 1);
  208. EXPECT_EQ(client.GetStatus(), "NOERROR");
  209. EXPECT_EQ(client.GetAnswer()[0].GetName(), "a.com");
  210. EXPECT_EQ(client.GetAnswer()[0].GetTTL(), 3);
  211. EXPECT_EQ(client.GetAnswer()[0].GetType(), "AAAA");
  212. EXPECT_EQ(client.GetAnswer()[0].GetData(), "2001:db8::1");
  213. }
  214. TEST_F(SubNet, per_server)
  215. {
  216. smartdns::MockServer server_upstream1;
  217. smartdns::MockServer server_upstream2;
  218. smartdns::Server server;
  219. server_upstream1.Start("udp://0.0.0.0:61053", [&](struct smartdns::ServerRequestContext *request) {
  220. if (request->qtype == DNS_T_A) {
  221. struct dns_opt_ecs ecs;
  222. struct dns_rrs *rrs = NULL;
  223. int rr_count = 0;
  224. int i = 0;
  225. int ret = 0;
  226. int has_ecs = 0;
  227. rr_count = 0;
  228. rrs = dns_get_rrs_start(request->packet, DNS_RRS_OPT, &rr_count);
  229. if (rr_count <= 0) {
  230. return smartdns::SERVER_REQUEST_ERROR;
  231. }
  232. for (i = 0; i < rr_count && rrs; i++, rrs = dns_get_rrs_next(request->packet, rrs)) {
  233. memset(&ecs, 0, sizeof(ecs));
  234. ret = dns_get_OPT_ECS(rrs, NULL, NULL, &ecs);
  235. if (ret != 0) {
  236. continue;
  237. }
  238. has_ecs = 1;
  239. break;
  240. }
  241. if (has_ecs == 1) {
  242. return smartdns::SERVER_REQUEST_ERROR;
  243. }
  244. smartdns::MockServer::AddIP(request, request->domain.c_str(), "1.2.3.4");
  245. return smartdns::SERVER_REQUEST_OK;
  246. }
  247. if (request->qtype == DNS_T_AAAA) {
  248. struct dns_opt_ecs ecs;
  249. struct dns_rrs *rrs = NULL;
  250. int rr_count = 0;
  251. int i = 0;
  252. int ret = 0;
  253. int has_ecs = 0;
  254. rr_count = 0;
  255. rrs = dns_get_rrs_start(request->packet, DNS_RRS_OPT, &rr_count);
  256. if (rr_count <= 0) {
  257. return smartdns::SERVER_REQUEST_ERROR;
  258. }
  259. for (i = 0; i < rr_count && rrs; i++, rrs = dns_get_rrs_next(request->packet, rrs)) {
  260. memset(&ecs, 0, sizeof(ecs));
  261. ret = dns_get_OPT_ECS(rrs, NULL, NULL, &ecs);
  262. if (ret != 0) {
  263. continue;
  264. }
  265. has_ecs = 1;
  266. break;
  267. }
  268. if (has_ecs == 1) {
  269. return smartdns::SERVER_REQUEST_ERROR;
  270. }
  271. smartdns::MockServer::AddIP(request, request->domain.c_str(), "2001:db8::1");
  272. return smartdns::SERVER_REQUEST_OK;
  273. }
  274. return smartdns::SERVER_REQUEST_SOA;
  275. });
  276. server_upstream2.Start("udp://0.0.0.0:62053", [&](struct smartdns::ServerRequestContext *request) {
  277. if (request->qtype == DNS_T_A) {
  278. struct dns_opt_ecs ecs;
  279. struct dns_rrs *rrs = NULL;
  280. int rr_count = 0;
  281. int i = 0;
  282. int ret = 0;
  283. int has_ecs = 0;
  284. rr_count = 0;
  285. rrs = dns_get_rrs_start(request->packet, DNS_RRS_OPT, &rr_count);
  286. if (rr_count <= 0) {
  287. return smartdns::SERVER_REQUEST_ERROR;
  288. }
  289. for (i = 0; i < rr_count && rrs; i++, rrs = dns_get_rrs_next(request->packet, rrs)) {
  290. memset(&ecs, 0, sizeof(ecs));
  291. ret = dns_get_OPT_ECS(rrs, NULL, NULL, &ecs);
  292. if (ret != 0) {
  293. continue;
  294. }
  295. has_ecs = 1;
  296. break;
  297. }
  298. if (has_ecs == 0) {
  299. smartdns::MockServer::AddIP(request, request->domain.c_str(), "1.2.3.4");
  300. return smartdns::SERVER_REQUEST_OK;
  301. }
  302. if (ecs.family != DNS_OPT_ECS_FAMILY_IPV4) {
  303. return smartdns::SERVER_REQUEST_ERROR;
  304. }
  305. if (memcmp(ecs.addr, "\x08\x08\x08\x00", 4) != 0) {
  306. return smartdns::SERVER_REQUEST_ERROR;
  307. }
  308. if (ecs.source_prefix != 24) {
  309. return smartdns::SERVER_REQUEST_ERROR;
  310. }
  311. smartdns::MockServer::AddIP(request, request->domain.c_str(), "5.6.7.8");
  312. return smartdns::SERVER_REQUEST_OK;
  313. }
  314. if (request->qtype = DNS_T_AAAA) {
  315. struct dns_opt_ecs ecs;
  316. struct dns_rrs *rrs = NULL;
  317. int rr_count = 0;
  318. int i = 0;
  319. int ret = 0;
  320. int has_ecs = 0;
  321. rr_count = 0;
  322. rrs = dns_get_rrs_start(request->packet, DNS_RRS_OPT, &rr_count);
  323. if (rr_count <= 0) {
  324. return smartdns::SERVER_REQUEST_ERROR;
  325. }
  326. for (i = 0; i < rr_count && rrs; i++, rrs = dns_get_rrs_next(request->packet, rrs)) {
  327. memset(&ecs, 0, sizeof(ecs));
  328. ret = dns_get_OPT_ECS(rrs, NULL, NULL, &ecs);
  329. if (ret != 0) {
  330. continue;
  331. }
  332. has_ecs = 1;
  333. break;
  334. }
  335. if (has_ecs == 0) {
  336. smartdns::MockServer::AddIP(request, request->domain.c_str(), "2001:db8::1");
  337. return smartdns::SERVER_REQUEST_ERROR;
  338. }
  339. if (ecs.family != DNS_OPT_ECS_FAMILY_IPV6) {
  340. return smartdns::SERVER_REQUEST_ERROR;
  341. }
  342. if (memcmp(ecs.addr, "\xff\xff\xff\xff\xff\xff\xff\xff\x00\x00\x00\x00\x00\x00\x00\x00", 16) != 0) {
  343. return smartdns::SERVER_REQUEST_ERROR;
  344. }
  345. if (ecs.source_prefix != 64) {
  346. return smartdns::SERVER_REQUEST_ERROR;
  347. }
  348. smartdns::MockServer::AddIP(request, request->domain.c_str(), "2001:db8::2");
  349. return smartdns::SERVER_REQUEST_OK;
  350. }
  351. return smartdns::SERVER_REQUEST_SOA;
  352. });
  353. server.MockPing(PING_TYPE_ICMP, "1.2.3.4", 60, 100);
  354. server.MockPing(PING_TYPE_ICMP, "5.6.7.8", 60, 10);
  355. server.MockPing(PING_TYPE_ICMP, "2001:db8::1", 60, 100);
  356. server.MockPing(PING_TYPE_ICMP, "2001:db8::2", 60, 10);
  357. server.Start(R"""(bind [::]:60053
  358. server 127.0.0.1:62053 -subnet=8.8.8.8/24 -subnet=ffff:ffff:ffff:ffff:ffff::/64
  359. server 127.0.0.1:61053
  360. log-num 0
  361. log-console yes
  362. dualstack-ip-selection no
  363. log-level debug
  364. cache-persist no)""");
  365. smartdns::Client client;
  366. ASSERT_TRUE(client.Query("a.com A", 60053));
  367. std::cout << client.GetResult() << std::endl;
  368. ASSERT_EQ(client.GetAnswerNum(), 1);
  369. EXPECT_EQ(client.GetStatus(), "NOERROR");
  370. EXPECT_EQ(client.GetAnswer()[0].GetName(), "a.com");
  371. EXPECT_EQ(client.GetAnswer()[0].GetTTL(), 3);
  372. EXPECT_EQ(client.GetAnswer()[0].GetType(), "A");
  373. EXPECT_EQ(client.GetAnswer()[0].GetData(), "5.6.7.8");
  374. ASSERT_TRUE(client.Query("a.com AAAA", 60053));
  375. std::cout << client.GetResult() << std::endl;
  376. ASSERT_EQ(client.GetAnswerNum(), 1);
  377. EXPECT_EQ(client.GetStatus(), "NOERROR");
  378. EXPECT_EQ(client.GetAnswer()[0].GetName(), "a.com");
  379. EXPECT_EQ(client.GetAnswer()[0].GetTTL(), 3);
  380. EXPECT_EQ(client.GetAnswer()[0].GetType(), "AAAA");
  381. EXPECT_EQ(client.GetAnswer()[0].GetData(), "2001:db8::2");
  382. }