Program.cs 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. using System;
  2. using System.Collections.Concurrent;
  3. using System.Collections.Generic;
  4. using System.Collections.ObjectModel;
  5. using System.IO;
  6. using System.Linq;
  7. using System.Net;
  8. using System.Net.Sockets;
  9. using System.Text;
  10. namespace DnsForwarder
  11. {
  12. internal class Program
  13. {
  14. private static string _defaultDns;
  15. private static string _cnDns;
  16. private static IReadOnlyDictionary<string, int> _chinaList;
  17. private static string[] _chinaListDns;
  18. private static Socket _listener;
  19. private static BlockingCollection<string> _consoleOutout;
  20. private static void Main(string[] args)
  21. {
  22. if (args.Length != 3)
  23. {
  24. Console.WriteLine("<default dns server ip> <.CN dns server ip> <path to dnsmasq-china-list file>");
  25. Environment.Exit(-1);
  26. return;
  27. }
  28. Console.WriteLine("Starting...");
  29. Console.WriteLine($"Default Server: {_defaultDns = args[0]}");
  30. Console.WriteLine($".CN DNS Server: {_cnDns = args[1]}");
  31. Console.WriteLine($"dnsmasq-china-list: {args[2]}");
  32. Console.WriteLine("Loading list file...");
  33. LoadListFile(args[2]);
  34. Console.WriteLine("OK. Listening...");
  35. _listener = new Socket(AddressFamily.InterNetwork, SocketType.Dgram, ProtocolType.Udp);
  36. _listener.Bind(new IPEndPoint(IPAddress.Any, 53));
  37. _consoleOutout = new BlockingCollection<string>();
  38. StartHandleRequest();
  39. while (true)
  40. {
  41. Console.WriteLine(_consoleOutout.Take());
  42. }
  43. }
  44. private static void StartHandleRequest()
  45. {
  46. var buf = new byte[1500];
  47. EndPoint from = new IPEndPoint(IPAddress.Any, 0);
  48. _listener.BeginReceiveFrom(buf, 0, buf.Length, SocketFlags.None, ref from, Callback, buf);
  49. }
  50. private static void Callback(IAsyncResult ar)
  51. {
  52. var flagNextStarted = false;
  53. EndPoint from = new IPEndPoint(IPAddress.Any, 0);
  54. string domain = null;
  55. string target = null;
  56. try
  57. {
  58. var count = _listener.EndReceiveFrom(ar, ref from);
  59. StartHandleRequest();
  60. flagNextStarted = true;
  61. var buf = (byte[])ar.AsyncState;
  62. domain = ExtractDomainName(buf);
  63. target = MatchServer(domain);
  64. var dnsResponse = GetDnsResponse(buf, count, target);
  65. if (dnsResponse != null) _listener.SendTo(dnsResponse, from);
  66. _consoleOutout.Add($"{DateTime.Now:yyyyMMdd HH:mm:ss} {from} [{target}]\t{domain}");
  67. }
  68. catch (Exception e)
  69. {
  70. _consoleOutout.Add($"{DateTime.Now:yyyyMMdd HH:mm:ss} {from} [{target ?? "Unknown"}]\t{domain ?? "Unknown"} Err:{e.Message}");
  71. }
  72. catch
  73. {
  74. _consoleOutout.Add($"{DateTime.Now:yyyyMMdd HH:mm:ss} {from} [{target ?? "Unknown"}]\t{domain ?? "Unknown"} Err:Unknown");
  75. }
  76. finally
  77. {
  78. if (flagNextStarted == false) StartHandleRequest();
  79. }
  80. }
  81. protected static string ExtractDomainName(byte[] buf)
  82. {
  83. //seeking for end of domain
  84. var ptr = 12;
  85. while (buf[ptr] != 0)
  86. {
  87. ptr += buf[ptr] + 1;
  88. }
  89. var bufDomain = new byte[ptr - 12];
  90. Array.Copy(buf, 12, bufDomain, 0, bufDomain.Length);
  91. //fill dots
  92. ptr = 0;
  93. while (ptr < bufDomain.Length)
  94. {
  95. var b = bufDomain[ptr];
  96. bufDomain[ptr] = (byte)'.';
  97. ptr += b + 1;
  98. }
  99. return Encoding.ASCII.GetString(bufDomain, 1, bufDomain.Length - 1);
  100. }
  101. protected static byte[] GetDnsResponse(byte[] buf, int count, string host, int port = 53)
  102. {
  103. using var to = new UdpClient();
  104. to.Connect(host, 53);
  105. to.Send(buf, count);
  106. //Handle Upstream TimeOut
  107. var asyncResult = to.BeginReceive(null, null);
  108. asyncResult.AsyncWaitHandle.WaitOne(2000);
  109. if (asyncResult.IsCompleted)
  110. {
  111. IPEndPoint remoteEP = null;
  112. byte[] receivedData = to.EndReceive(asyncResult, ref remoteEP);
  113. return receivedData;
  114. }
  115. return null;
  116. }
  117. protected static string MatchServer(string domain)
  118. {
  119. var lower = domain.ToLower();
  120. if (lower.EndsWith(".cn")) return _cnDns;
  121. var parts = lower.Split('.').Reverse().ToArray();
  122. for (int i = parts.Length - 1; i >= 0; i--)
  123. {
  124. var d = string.Join(".", parts.Take(i + 1).Reverse());
  125. if (_chinaList.TryGetValue(d, out var tar)) return _chinaListDns[tar];
  126. }
  127. return _defaultDns;
  128. }
  129. protected static void LoadListFile(params string[] paths)
  130. {
  131. var lines = paths.SelectMany(File.ReadAllLines);
  132. var dic = new Dictionary<string, int>();
  133. var tar = new List<string>();
  134. foreach (var line in lines)
  135. {
  136. var p = line.Trim();
  137. if (p.StartsWith("#")) continue;
  138. var parts = p.Split('/');
  139. if (parts.Length != 3 && parts[0] != "server=") continue;
  140. var domain = parts[1];
  141. var dns = parts[2];
  142. var dnsIndex = tar.IndexOf(dns);
  143. if (dnsIndex == -1)
  144. {
  145. tar.Add(dns);
  146. dnsIndex = tar.Count - 1;
  147. }
  148. dic[domain] = dnsIndex;
  149. }
  150. _chinaListDns = tar.ToArray();
  151. _chinaList = new ReadOnlyDictionary<string, int>(dic);
  152. }
  153. }
  154. }