Skip to content

Commit

Permalink
dns服务器状态缓存
Browse files Browse the repository at this point in the history
  • Loading branch information
xljiulang committed Dec 3, 2021
1 parent 1685a5c commit c51cda0
Showing 1 changed file with 52 additions and 16 deletions.
68 changes: 52 additions & 16 deletions FastGithub.DomainResolve/DnsClient.cs
Original file line number Diff line number Diff line change
Expand Up @@ -31,13 +31,15 @@ sealed class DnsClient
private readonly ILogger<DnsClient> logger;

private readonly ConcurrentDictionary<string, SemaphoreSlim> semaphoreSlims = new();
private readonly IMemoryCache dnsStateCache = new MemoryCache(Options.Create(new MemoryCacheOptions()));
private readonly IMemoryCache dnsLookupCache = new MemoryCache(Options.Create(new MemoryCacheOptions()));

private readonly TimeSpan stateExpiration = TimeSpan.FromMinutes(5d);
private readonly TimeSpan minTimeToLive = TimeSpan.FromSeconds(30d);
private readonly TimeSpan maxTimeToLive = TimeSpan.FromMinutes(10d);

private readonly int resolveTimeout = (int)TimeSpan.FromSeconds(4d).TotalMilliseconds;
private static readonly TimeSpan connectTimeout = TimeSpan.FromSeconds(2d);
private static readonly TimeSpan tcpConnectTimeout = TimeSpan.FromSeconds(2d);

private record LookupResult(IList<IPAddress> Addresses, TimeSpan TimeToLive);

Expand Down Expand Up @@ -67,7 +69,7 @@ public DnsClient(
public async IAsyncEnumerable<IPAddress> ResolveAsync(DnsEndPoint endPoint, bool fastSort, [EnumeratorCancellation] CancellationToken cancellationToken)
{
var hashSet = new HashSet<IPAddress>();
foreach (var dns in this.GetDnsServers())
await foreach (var dns in this.GetDnsServersAsync(cancellationToken))
{
var addresses = await this.LookupAsync(dns, endPoint, fastSort, cancellationToken);
foreach (var address in addresses)
Expand All @@ -84,7 +86,7 @@ public async IAsyncEnumerable<IPAddress> ResolveAsync(DnsEndPoint endPoint, bool
/// 获取dns服务
/// </summary>
/// <returns></returns>
private IEnumerable<IPEndPoint> GetDnsServers()
private async IAsyncEnumerable<IPEndPoint> GetDnsServersAsync([EnumeratorCancellation] CancellationToken cancellationToken)
{
var cryptDns = this.dnscryptProxy.LocalEndPoint;
if (cryptDns != null)
Expand All @@ -93,15 +95,52 @@ private IEnumerable<IPEndPoint> GetDnsServers()
yield return cryptDns;
}

foreach (var fallbackDns in this.fastGithubConfig.FallbackDns)
foreach (var dns in this.fastGithubConfig.FallbackDns)
{
if (Socket.OSSupportsIPv6 || fallbackDns.AddressFamily != AddressFamily.InterNetworkV6)
if (await this.IsDnsAvailableAsync(dns, cancellationToken))
{
yield return fallbackDns;
yield return dns;
}
}
}

/// <summary>
/// 获取dns是否可用
/// </summary>
/// <param name="dns"></param>
/// <param name="cancellationToken"></param>
/// <returns></returns>
private async ValueTask<bool> IsDnsAvailableAsync(IPEndPoint dns, CancellationToken cancellationToken)
{
if (dns.Port != DNS_PORT)
{
return true;
}

if (this.dnsStateCache.TryGetValue<bool>(dns, out var state))
{
return state;
}

var key = dns.ToString();
var semaphore = this.semaphoreSlims.GetOrAdd(key, _ => new SemaphoreSlim(1, 1));
await semaphore.WaitAsync(CancellationToken.None);

try
{
using var timeoutTokenSource = new CancellationTokenSource(tcpConnectTimeout);
using var linkedTokenSource = CancellationTokenSource.CreateLinkedTokenSource(timeoutTokenSource.Token, cancellationToken);
using var socket = new Socket(dns.AddressFamily, SocketType.Stream, ProtocolType.Tcp);
await socket.ConnectAsync(dns, linkedTokenSource.Token);
return this.dnsStateCache.Set(dns, true, this.stateExpiration);
}
catch (Exception)
{
cancellationToken.ThrowIfCancellationRequested();
return this.dnsStateCache.Set(dns, false, this.stateExpiration);
}
}

/// <summary>
/// 解析域名
/// </summary>
Expand Down Expand Up @@ -132,7 +171,7 @@ private async Task<IList<IPAddress>> LookupAsync(IPEndPoint dns, DnsEndPoint end
catch (Exception ex)
{
this.logger.LogWarning($"{endPoint.Host}@{dns}->{ex.Message}");
var expiration = IsTcpResetException(ex) ? this.maxTimeToLive : this.minTimeToLive;
var expiration = IsSocketException(ex) ? this.maxTimeToLive : this.minTimeToLive;
return this.dnsLookupCache.Set(key, Array.Empty<IPAddress>(), expiration);
}
finally
Expand All @@ -142,22 +181,19 @@ private async Task<IList<IPAddress>> LookupAsync(IPEndPoint dns, DnsEndPoint end
}

/// <summary>
/// 是否为因收到tcp reset导致的关闭
/// 是否为Socket异常
/// </summary>
/// <param name="ex"></param>
/// <returns></returns>
private static bool IsTcpResetException(Exception ex)
private static bool IsSocketException(Exception ex)
{
if (ex is SocketException socketException)
if (ex is SocketException)
{
if (socketException.SocketErrorCode == SocketError.ConnectionReset)
{
return true;
}
return true;
}

var inner = ex.InnerException;
return inner != null && IsTcpResetException(inner);
return inner != null && IsSocketException(inner);
}


Expand Down Expand Up @@ -272,7 +308,7 @@ private static async Task<IList<IPAddress>> OrderByConnectAnyAsync(IList<IPAddre
return addresses;
}

using var controlTokenSource = new CancellationTokenSource(connectTimeout);
using var controlTokenSource = new CancellationTokenSource(tcpConnectTimeout);
using var linkedTokenSource = CancellationTokenSource.CreateLinkedTokenSource(cancellationToken, controlTokenSource.Token);

var connectTasks = addresses.Select(address => ConnectAsync(address, port, linkedTokenSource.Token));
Expand Down

0 comments on commit c51cda0

Please sign in to comment.