DomainResolver.cs 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. using DNS.Client;
  2. using DNS.Protocol;
  3. using FastGithub.Configuration;
  4. using Microsoft.Extensions.Caching.Memory;
  5. using Microsoft.Extensions.Logging;
  6. using Microsoft.Extensions.Options;
  7. using System;
  8. using System.Collections.Concurrent;
  9. using System.Collections.Generic;
  10. using System.Linq;
  11. using System.Net;
  12. using System.Net.Sockets;
  13. using System.Threading;
  14. using System.Threading.Tasks;
  15. namespace FastGithub.DomainResolve
  16. {
  17. /// <summary>
  18. /// 域名解析器
  19. /// </summary>
  20. sealed class DomainResolver : IDomainResolver
  21. {
  22. private readonly IMemoryCache domainResolveCache = new MemoryCache(Options.Create(new MemoryCacheOptions()));
  23. private readonly IMemoryCache disableIPAddressCache = new MemoryCache(Options.Create(new MemoryCacheOptions()));
  24. private readonly FastGithubConfig fastGithubConfig;
  25. private readonly DnscryptProxy dnscryptProxy;
  26. private readonly ILogger<DomainResolver> logger;
  27. private readonly TimeSpan connectTimeout = TimeSpan.FromSeconds(5d);
  28. private readonly TimeSpan disableIPExpiration = TimeSpan.FromMinutes(2d);
  29. private readonly TimeSpan dnscryptExpiration = TimeSpan.FromMinutes(10d);
  30. private readonly TimeSpan fallbackExpiration = TimeSpan.FromMinutes(2d);
  31. private readonly TimeSpan loopbackExpiration = TimeSpan.FromSeconds(5d);
  32. private readonly ConcurrentDictionary<DnsEndPoint, SemaphoreSlim> semaphoreSlims = new();
  33. /// <summary>
  34. /// 域名解析器
  35. /// </summary>
  36. /// <param name="fastGithubConfig"></param>
  37. /// <param name="dnscryptProxy"></param>
  38. /// <param name="logger"></param>
  39. public DomainResolver(
  40. FastGithubConfig fastGithubConfig,
  41. DnscryptProxy dnscryptProxy,
  42. ILogger<DomainResolver> logger)
  43. {
  44. this.fastGithubConfig = fastGithubConfig;
  45. this.dnscryptProxy = dnscryptProxy;
  46. this.logger = logger;
  47. }
  48. /// <summary>
  49. /// 设置ip不可用
  50. /// </summary>
  51. /// <param name="address">ip</param>
  52. public void SetDisabled(IPAddress address)
  53. {
  54. this.disableIPAddressCache.Set(address, address, this.disableIPExpiration);
  55. }
  56. /// <summary>
  57. /// 刷新域名解析结果
  58. /// </summary>
  59. /// <param name="domain">域名</param>
  60. public void FlushDomain(DnsEndPoint domain)
  61. {
  62. this.domainResolveCache.Remove(domain);
  63. }
  64. /// <summary>
  65. /// 解析域名
  66. /// </summary>
  67. /// <param name="domain"></param>
  68. /// <param name="cancellationToken"></param>
  69. /// <returns></returns>
  70. public async Task<IPAddress> ResolveAsync(DnsEndPoint domain, CancellationToken cancellationToken = default)
  71. {
  72. var semaphore = this.semaphoreSlims.GetOrAdd(domain, _ => new SemaphoreSlim(1, 1));
  73. try
  74. {
  75. await semaphore.WaitAsync();
  76. for (var i = 0; i < 2; i++)
  77. {
  78. var address = await this.ResolveCoreAsync(domain, cancellationToken);
  79. if (address != null)
  80. {
  81. return address;
  82. }
  83. }
  84. throw new FastGithubException($"当前解析不到{domain.Host}可用的ip,请刷新重试");
  85. }
  86. finally
  87. {
  88. semaphore.Release();
  89. }
  90. }
  91. /// <summary>
  92. /// 解析域名
  93. /// </summary>
  94. /// <param name="domain"></param>
  95. /// <param name="cancellationToken"></param>
  96. /// <returns></returns>
  97. private async Task<IPAddress?> ResolveCoreAsync(DnsEndPoint domain, CancellationToken cancellationToken)
  98. {
  99. if (this.domainResolveCache.TryGetValue<IPAddress>(domain, out var address) && address != null)
  100. {
  101. return address;
  102. }
  103. var expiration = this.dnscryptExpiration;
  104. if (this.dnscryptProxy.LocalEndPoint != null)
  105. {
  106. address = await this.LookupAsync(this.dnscryptProxy.LocalEndPoint, domain, cancellationToken);
  107. }
  108. if (address == null)
  109. {
  110. expiration = this.fallbackExpiration;
  111. address = await this.LookupByFallbackAsync(domain, cancellationToken);
  112. }
  113. if (address == null)
  114. {
  115. return null;
  116. }
  117. // 往往是被污染的dns
  118. if (address.Equals(IPAddress.Loopback) == true)
  119. {
  120. expiration = this.loopbackExpiration;
  121. }
  122. this.domainResolveCache.Set(domain, address, expiration);
  123. return address;
  124. }
  125. /// <summary>
  126. /// 回退查找ip
  127. /// </summary>
  128. /// <param name="domain"></param>
  129. /// <param name="cancellationToken"></param>
  130. /// <returns></returns>
  131. private async Task<IPAddress?> LookupByFallbackAsync(DnsEndPoint domain, CancellationToken cancellationToken)
  132. {
  133. foreach (var dns in this.fastGithubConfig.FallbackDns)
  134. {
  135. var address = await this.LookupAsync(dns, domain, cancellationToken);
  136. if (address != null)
  137. {
  138. return address;
  139. }
  140. }
  141. return default;
  142. }
  143. /// <summary>
  144. /// 查找最快的可用ip
  145. /// </summary>
  146. /// <param name="dns"></param>
  147. /// <param name="domain"></param>
  148. /// <param name="cancellationToken"></param>
  149. /// <returns></returns>
  150. private async Task<IPAddress?> LookupAsync(IPEndPoint dns, DnsEndPoint domain, CancellationToken cancellationToken)
  151. {
  152. try
  153. {
  154. var dnsClient = new DnsClient(dns);
  155. var addresses = await dnsClient.Lookup(domain.Host, RecordType.A, cancellationToken);
  156. addresses = addresses.Where(address => this.disableIPAddressCache.TryGetValue(address, out _) == false).ToList();
  157. var address = await this.FindFastValueAsync(addresses, domain.Port, cancellationToken);
  158. if (address == null)
  159. {
  160. this.logger.LogWarning($"dns({dns})解析不到{domain.Host}可用的ip解析");
  161. }
  162. else
  163. {
  164. this.logger.LogInformation($"dns({dns}): {domain.Host}->{address}");
  165. }
  166. return address;
  167. }
  168. catch (Exception ex)
  169. {
  170. this.logger.LogWarning($"dns({dns})无法解析{domain.Host}:{ex.Message}");
  171. return default;
  172. }
  173. }
  174. /// <summary>
  175. /// 获取最快的ip
  176. /// </summary>
  177. /// <param name="addresses"></param>
  178. /// <param name="port"></param>
  179. /// <param name="cancellationToken"></param>
  180. /// <returns></returns>
  181. private async Task<IPAddress?> FindFastValueAsync(IEnumerable<IPAddress> addresses, int port, CancellationToken cancellationToken)
  182. {
  183. if (addresses.Any() == false)
  184. {
  185. return default;
  186. }
  187. if (port <= 0)
  188. {
  189. return addresses.FirstOrDefault();
  190. }
  191. var tasks = addresses.Select(address => this.IsAvailableAsync(address, port, cancellationToken));
  192. var fastTask = await Task.WhenAny(tasks);
  193. return await fastTask;
  194. }
  195. /// <summary>
  196. /// 验证远程节点是否可连接
  197. /// </summary>
  198. /// <param name="address"></param>
  199. /// <param name="port"></param>
  200. /// <param name="cancellationToken"></param>
  201. /// <returns></returns>
  202. private async Task<IPAddress?> IsAvailableAsync(IPAddress address, int port, CancellationToken cancellationToken)
  203. {
  204. try
  205. {
  206. using var socket = new Socket(SocketType.Stream, ProtocolType.Tcp);
  207. using var timeoutTokenSource = new CancellationTokenSource(this.connectTimeout);
  208. using var linkedTokenSource = CancellationTokenSource.CreateLinkedTokenSource(cancellationToken, timeoutTokenSource.Token);
  209. await socket.ConnectAsync(address, port, linkedTokenSource.Token);
  210. return address;
  211. }
  212. catch (OperationCanceledException)
  213. {
  214. this.SetDisabled(address);
  215. return default;
  216. }
  217. catch (Exception)
  218. {
  219. this.SetDisabled(address);
  220. await Task.Delay(this.connectTimeout, cancellationToken);
  221. return default;
  222. }
  223. }
  224. }
  225. }