Program.cs 6.2 KB

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