DomainResolver.cs 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  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 blackIPAddressCache = new MemoryCache(Options.Create(new MemoryCacheOptions()));
  23. private readonly IMemoryCache domainResolveCache = 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 lookupTimeout = TimeSpan.FromSeconds(5d);
  28. private readonly TimeSpan connectTimeout = TimeSpan.FromSeconds(5d);
  29. private readonly TimeSpan dnscryptExpiration = TimeSpan.FromMinutes(5d);
  30. private readonly TimeSpan fallbackExpiration = TimeSpan.FromMinutes(1d);
  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. /// <param name="expiration">过期时间</param>
  53. public void SetBlack(IPAddress address, TimeSpan expiration)
  54. {
  55. this.blackIPAddressCache.Set(address, address, expiration);
  56. this.logger.LogWarning($"已将{address}关到小黑屋{expiration.TotalMinutes}分钟");
  57. }
  58. /// <summary>
  59. /// 刷新域名解析结果
  60. /// </summary>
  61. /// <param name="domain">域名</param>
  62. public void FlushDomain(DnsEndPoint domain)
  63. {
  64. this.domainResolveCache.Remove(domain);
  65. }
  66. /// <summary>
  67. /// 解析域名
  68. /// </summary>
  69. /// <param name="domain"></param>
  70. /// <param name="cancellationToken"></param>
  71. /// <returns></returns>
  72. public async Task<IPAddress> ResolveAsync(DnsEndPoint domain, CancellationToken cancellationToken = default)
  73. {
  74. var semaphore = this.semaphoreSlims.GetOrAdd(domain, _ => new SemaphoreSlim(1, 1));
  75. try
  76. {
  77. await semaphore.WaitAsync();
  78. return await this.LookupAsync(domain, cancellationToken);
  79. }
  80. finally
  81. {
  82. semaphore.Release();
  83. }
  84. }
  85. /// <summary>
  86. /// 查找ip
  87. /// </summary>
  88. /// <param name="domain"></param>
  89. /// <param name="cancellationToken"></param>
  90. /// <returns></returns>
  91. private async Task<IPAddress> LookupAsync(DnsEndPoint domain, CancellationToken cancellationToken)
  92. {
  93. if (this.domainResolveCache.TryGetValue<IPAddress>(domain, out var address))
  94. {
  95. return address;
  96. }
  97. var expiration = this.dnscryptExpiration;
  98. if (this.dnscryptProxy.LocalEndPoint != null)
  99. {
  100. address = await this.LookupCoreAsync(this.dnscryptProxy.LocalEndPoint, domain, cancellationToken);
  101. }
  102. if (address == null)
  103. {
  104. expiration = this.fallbackExpiration;
  105. address = await this.FallbackLookupAsync(domain, cancellationToken);
  106. }
  107. if (address == null)
  108. {
  109. throw new FastGithubException($"当前解析不到{domain.Host}可用的ip,请刷新重试");
  110. }
  111. // 往往是被污染的dns
  112. if (address.Equals(IPAddress.Loopback) == true)
  113. {
  114. expiration = this.loopbackExpiration;
  115. }
  116. this.logger.LogInformation($"[{domain.Host}->{address}]");
  117. this.domainResolveCache.Set(domain, address, expiration);
  118. return address;
  119. }
  120. /// <summary>
  121. /// 回退查找ip
  122. /// </summary>
  123. /// <param name="domain"></param>
  124. /// <param name="cancellationToken"></param>
  125. /// <returns></returns>
  126. private async Task<IPAddress?> FallbackLookupAsync(DnsEndPoint domain, CancellationToken cancellationToken)
  127. {
  128. foreach (var dns in this.fastGithubConfig.FallbackDns)
  129. {
  130. var address = await this.LookupCoreAsync(dns, domain, cancellationToken);
  131. if (address != null)
  132. {
  133. return address;
  134. }
  135. }
  136. return default;
  137. }
  138. /// <summary>
  139. /// 查找ip
  140. /// </summary>
  141. /// <param name="dns"></param>
  142. /// <param name="domain"></param>
  143. /// <param name="cancellationToken"></param>
  144. /// <returns></returns>
  145. private async Task<IPAddress?> LookupCoreAsync(IPEndPoint dns, DnsEndPoint domain, CancellationToken cancellationToken)
  146. {
  147. try
  148. {
  149. var dnsClient = new DnsClient(dns);
  150. using var timeoutTokenSource = new CancellationTokenSource(this.lookupTimeout);
  151. using var linkedTokenSource = CancellationTokenSource.CreateLinkedTokenSource(cancellationToken, timeoutTokenSource.Token);
  152. var addresses = await dnsClient.Lookup(domain.Host, RecordType.A, linkedTokenSource.Token);
  153. return await this.FindFastValueAsync(addresses, domain.Port, cancellationToken);
  154. }
  155. catch (Exception ex)
  156. {
  157. this.logger.LogWarning($"dns({dns})无法解析{domain.Host}:{ex.Message}");
  158. return default;
  159. }
  160. }
  161. /// <summary>
  162. /// 获取最快的ip
  163. /// </summary>
  164. /// <param name="addresses"></param>
  165. /// <param name="port"></param>
  166. /// <param name="cancellationToken"></param>
  167. /// <returns></returns>
  168. private async Task<IPAddress?> FindFastValueAsync(IEnumerable<IPAddress> addresses, int port, CancellationToken cancellationToken)
  169. {
  170. if (addresses.Any() == false)
  171. {
  172. return default;
  173. }
  174. var tasks = addresses.Select(address => this.IsAvailableAsync(address, port, cancellationToken));
  175. var fastTask = await Task.WhenAny(tasks);
  176. return await fastTask;
  177. }
  178. /// <summary>
  179. /// 验证远程节点是否可连接
  180. /// </summary>
  181. /// <param name="address"></param>
  182. /// <param name="port"></param>
  183. /// <param name="cancellationToken"></param>
  184. /// <returns></returns>
  185. private async Task<IPAddress?> IsAvailableAsync(IPAddress address, int port, CancellationToken cancellationToken)
  186. {
  187. if (port <= 0)
  188. {
  189. return address;
  190. }
  191. if (this.blackIPAddressCache.TryGetValue(address, out _))
  192. {
  193. this.logger.LogWarning($"已跳过黑名单IP:{address}");
  194. return default;
  195. }
  196. try
  197. {
  198. using var socket = new Socket(SocketType.Stream, ProtocolType.Tcp);
  199. using var timeoutTokenSource = new CancellationTokenSource(this.connectTimeout);
  200. using var linkedTokenSource = CancellationTokenSource.CreateLinkedTokenSource(cancellationToken, timeoutTokenSource.Token);
  201. await socket.ConnectAsync(address, port, linkedTokenSource.Token);
  202. return address;
  203. }
  204. catch (OperationCanceledException)
  205. {
  206. this.logger.LogWarning($"已忽略连接超时的IP:{address}");
  207. return default;
  208. }
  209. catch (Exception)
  210. {
  211. this.logger.LogWarning($"已忽略不可连接的IP:{address}");
  212. await Task.Delay(this.connectTimeout, cancellationToken);
  213. return default;
  214. }
  215. }
  216. }
  217. }