Introduce DnsCache API + DnsResolver extensibility
Motivation: Caching is currently nested in DnsResolver. It should also be possible to extend DnsResolver to ba able to pass a different cache on each resolution attemp. Modifications: * Introduce DnsCache, NoopDnsCache and DefaultDnsCache. The latter contains all the current caching logic that was extracted. * Introduce protected versions of doResolve and doResolveAll that can be used as extension points to build resolvers that bypass the main cache and use a different one on each resolution. Result: Isolated caching logic. Better extensibility.
This commit is contained in:
parent
07d0bcab8c
commit
b4be040f30
@ -76,4 +76,24 @@ public final class ObjectUtil {
|
|||||||
checkPositive(array.length, name + ".length");
|
checkPositive(array.length, name + ".length");
|
||||||
return array;
|
return array;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Resolves a possibly null Integer to a primitive int, using a default value.
|
||||||
|
* @param wrapper the wrapper
|
||||||
|
* @param defaultValue the default value
|
||||||
|
* @return the primitive value
|
||||||
|
*/
|
||||||
|
public static int intValue(Integer wrapper, int defaultValue) {
|
||||||
|
return wrapper != null ? wrapper.intValue() : defaultValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Resolves a possibly null Long to a primitive long, using a default value.
|
||||||
|
* @param wrapper the wrapper
|
||||||
|
* @param defaultValue the default value
|
||||||
|
* @return the primitive value
|
||||||
|
*/
|
||||||
|
public static long longValue(Long wrapper, long defaultValue) {
|
||||||
|
return wrapper != null ? wrapper.longValue() : defaultValue;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,222 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2016 The Netty Project
|
||||||
|
*
|
||||||
|
* The Netty Project licenses this file to you under the Apache License,
|
||||||
|
* version 2.0 (the "License"); you may not use this file except in compliance
|
||||||
|
* with the License. You may obtain a copy of the License at:
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
package io.netty.resolver.dns;
|
||||||
|
|
||||||
|
import io.netty.channel.EventLoop;
|
||||||
|
import io.netty.util.internal.OneTimeTask;
|
||||||
|
import io.netty.util.internal.PlatformDependent;
|
||||||
|
|
||||||
|
import java.net.InetAddress;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.concurrent.ConcurrentMap;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
import static io.netty.util.internal.ObjectUtil.checkNotNull;
|
||||||
|
import static io.netty.util.internal.ObjectUtil.checkPositiveOrZero;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Default implementation of {@link DnsCache}, backed by a {@link ConcurrentMap}.
|
||||||
|
*/
|
||||||
|
public class DefaultDnsCache implements DnsCache {
|
||||||
|
|
||||||
|
private final ConcurrentMap<String, List<DnsCacheEntry>> resolveCache = PlatformDependent.newConcurrentHashMap();
|
||||||
|
private final int minTtl;
|
||||||
|
private final int maxTtl;
|
||||||
|
private final int negativeTtl;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a cache that respects the TTL returned by the DNS server
|
||||||
|
* and doesn't cache negative responses.
|
||||||
|
*/
|
||||||
|
public DefaultDnsCache() {
|
||||||
|
this(0, Integer.MAX_VALUE, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a cache.
|
||||||
|
* @param minTtl the minimum TTL
|
||||||
|
* @param maxTtl the maximum TTL
|
||||||
|
* @param negativeTtl the TTL for failed queries
|
||||||
|
*/
|
||||||
|
public DefaultDnsCache(int minTtl, int maxTtl, int negativeTtl) {
|
||||||
|
this.minTtl = checkPositiveOrZero(minTtl, "minTtl");
|
||||||
|
this.maxTtl = checkPositiveOrZero(maxTtl, "maxTtl");
|
||||||
|
if (minTtl > maxTtl) {
|
||||||
|
throw new IllegalArgumentException(
|
||||||
|
"minTtl: " + minTtl + ", maxTtl: " + maxTtl + " (expected: 0 <= minTtl <= maxTtl)");
|
||||||
|
}
|
||||||
|
this.negativeTtl = checkPositiveOrZero(negativeTtl, "negativeTtl");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the minimum TTL of the cached DNS resource records (in seconds).
|
||||||
|
*
|
||||||
|
* @see #maxTtl()
|
||||||
|
*/
|
||||||
|
public int minTtl() {
|
||||||
|
return minTtl;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the maximum TTL of the cached DNS resource records (in seconds).
|
||||||
|
*
|
||||||
|
* @see #minTtl()
|
||||||
|
*/
|
||||||
|
public int maxTtl() {
|
||||||
|
return maxTtl;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the TTL of the cache for the failed DNS queries (in seconds). The default value is {@code 0}, which
|
||||||
|
* disables the cache for negative results.
|
||||||
|
*/
|
||||||
|
public int negativeTtl() {
|
||||||
|
return negativeTtl;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void clear() {
|
||||||
|
for (Iterator<Map.Entry<String, List<DnsCacheEntry>>> i = resolveCache.entrySet().iterator(); i.hasNext();) {
|
||||||
|
final Map.Entry<String, List<DnsCacheEntry>> e = i.next();
|
||||||
|
i.remove();
|
||||||
|
cancelExpiration(e.getValue());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean clear(String hostname) {
|
||||||
|
checkNotNull(hostname, "hostname");
|
||||||
|
boolean removed = false;
|
||||||
|
for (Iterator<Map.Entry<String, List<DnsCacheEntry>>> i = resolveCache.entrySet().iterator(); i.hasNext();) {
|
||||||
|
final Map.Entry<String, List<DnsCacheEntry>> e = i.next();
|
||||||
|
if (e.getKey().equals(hostname)) {
|
||||||
|
i.remove();
|
||||||
|
cancelExpiration(e.getValue());
|
||||||
|
removed = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return removed;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<DnsCacheEntry> get(String hostname) {
|
||||||
|
checkNotNull(hostname, "hostname");
|
||||||
|
return resolveCache.get(hostname);
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<DnsCacheEntry> cachedEntries(String hostname) {
|
||||||
|
List<DnsCacheEntry> oldEntries = resolveCache.get(hostname);
|
||||||
|
final List<DnsCacheEntry> entries;
|
||||||
|
if (oldEntries == null) {
|
||||||
|
List<DnsCacheEntry> newEntries = new ArrayList<DnsCacheEntry>(8);
|
||||||
|
oldEntries = resolveCache.putIfAbsent(hostname, newEntries);
|
||||||
|
entries = oldEntries != null? oldEntries : newEntries;
|
||||||
|
} else {
|
||||||
|
entries = oldEntries;
|
||||||
|
}
|
||||||
|
return entries;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void cache(String hostname, InetAddress address, long originalTtl, EventLoop loop) {
|
||||||
|
if (maxTtl == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
checkNotNull(hostname, "hostname");
|
||||||
|
checkNotNull(address, "address");
|
||||||
|
checkNotNull(loop, "loop");
|
||||||
|
|
||||||
|
final int ttl = Math.max(minTtl, (int) Math.min(maxTtl, originalTtl));
|
||||||
|
final List<DnsCacheEntry> entries = cachedEntries(hostname);
|
||||||
|
final DnsCacheEntry e = new DnsCacheEntry(hostname, address);
|
||||||
|
|
||||||
|
synchronized (entries) {
|
||||||
|
if (!entries.isEmpty()) {
|
||||||
|
final DnsCacheEntry firstEntry = entries.get(0);
|
||||||
|
if (firstEntry.cause() != null) {
|
||||||
|
assert entries.size() == 1;
|
||||||
|
firstEntry.cancelExpiration();
|
||||||
|
entries.clear();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
entries.add(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
scheduleCacheExpiration(entries, e, ttl, loop);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void cache(String hostname, Throwable cause, EventLoop loop) {
|
||||||
|
if (negativeTtl == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
checkNotNull(hostname, "hostname");
|
||||||
|
checkNotNull(cause, "cause");
|
||||||
|
checkNotNull(loop, "loop");
|
||||||
|
|
||||||
|
final List<DnsCacheEntry> entries = cachedEntries(hostname);
|
||||||
|
final DnsCacheEntry e = new DnsCacheEntry(hostname, cause);
|
||||||
|
|
||||||
|
synchronized (entries) {
|
||||||
|
final int numEntries = entries.size();
|
||||||
|
for (int i = 0; i < numEntries; i ++) {
|
||||||
|
entries.get(i).cancelExpiration();
|
||||||
|
}
|
||||||
|
entries.clear();
|
||||||
|
entries.add(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
scheduleCacheExpiration(entries, e, negativeTtl, loop);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void cancelExpiration(List<DnsCacheEntry> entries) {
|
||||||
|
final int numEntries = entries.size();
|
||||||
|
for (int i = 0; i < numEntries; i++) {
|
||||||
|
entries.get(i).cancelExpiration();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void scheduleCacheExpiration(final List<DnsCacheEntry> entries,
|
||||||
|
final DnsCacheEntry e,
|
||||||
|
int ttl,
|
||||||
|
EventLoop loop) {
|
||||||
|
e.scheduleExpiration(loop, new OneTimeTask() {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
synchronized (entries) {
|
||||||
|
entries.remove(e);
|
||||||
|
if (entries.isEmpty()) {
|
||||||
|
resolveCache.remove(e.hostname());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, ttl, TimeUnit.SECONDS);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return new StringBuilder()
|
||||||
|
.append("DefaultDnsCache(minTtl=")
|
||||||
|
.append(minTtl).append(", maxTtl=")
|
||||||
|
.append(maxTtl).append(", negativeTtl=")
|
||||||
|
.append(negativeTtl).append(", cached resolved hostname=")
|
||||||
|
.append(resolveCache.size()).append(")")
|
||||||
|
.toString();
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,68 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2016 The Netty Project
|
||||||
|
*
|
||||||
|
* The Netty Project licenses this file to you under the Apache License,
|
||||||
|
* version 2.0 (the "License"); you may not use this file except in compliance
|
||||||
|
* with the License. You may obtain a copy of the License at:
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
package io.netty.resolver.dns;
|
||||||
|
|
||||||
|
import io.netty.channel.EventLoop;
|
||||||
|
|
||||||
|
import java.net.InetAddress;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A cache for DNS resolution entries.
|
||||||
|
*/
|
||||||
|
public interface DnsCache {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clears all the resolved addresses cached by this resolver.
|
||||||
|
*
|
||||||
|
* @return {@code this}
|
||||||
|
*
|
||||||
|
* @see #clear(String)
|
||||||
|
*/
|
||||||
|
void clear();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clears the resolved addresses of the specified host name from the cache of this resolver.
|
||||||
|
*
|
||||||
|
* @return {@code true} if and only if there was an entry for the specified host name in the cache and
|
||||||
|
* it has been removed by this method
|
||||||
|
*/
|
||||||
|
boolean clear(String hostname);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the cached entries for the given hostname.
|
||||||
|
* @param hostname the hostname
|
||||||
|
* @return the cached entries
|
||||||
|
*/
|
||||||
|
List<DnsCacheEntry> get(String hostname);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cache a resolved address for a given hostname.
|
||||||
|
* @param hostname the hostname
|
||||||
|
* @param address the resolved adresse
|
||||||
|
* @param originalTtl the TLL as returned by the DNS server
|
||||||
|
* @param loop the {@link EventLoop} used to register the TTL timeout
|
||||||
|
*/
|
||||||
|
void cache(String hostname, InetAddress address, long originalTtl, EventLoop loop);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cache the resolution failure for a given hostname.
|
||||||
|
* @param hostname the hostname
|
||||||
|
* @param cause the resolution failure
|
||||||
|
* @param loop the {@link EventLoop} used to register the TTL timeout
|
||||||
|
*/
|
||||||
|
void cache(String hostname, Throwable cause, EventLoop loop);
|
||||||
|
}
|
@ -16,40 +16,45 @@
|
|||||||
|
|
||||||
package io.netty.resolver.dns;
|
package io.netty.resolver.dns;
|
||||||
|
|
||||||
|
import static io.netty.util.internal.ObjectUtil.checkNotNull;
|
||||||
|
|
||||||
import io.netty.channel.EventLoop;
|
import io.netty.channel.EventLoop;
|
||||||
import io.netty.util.concurrent.ScheduledFuture;
|
import io.netty.util.concurrent.ScheduledFuture;
|
||||||
|
|
||||||
import java.net.InetAddress;
|
import java.net.InetAddress;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
final class DnsCacheEntry {
|
/**
|
||||||
|
* Entry in {@link DnsCache}.
|
||||||
|
*/
|
||||||
|
public final class DnsCacheEntry {
|
||||||
|
|
||||||
private final String hostname;
|
private final String hostname;
|
||||||
private final InetAddress address;
|
private final InetAddress address;
|
||||||
private final Throwable cause;
|
private final Throwable cause;
|
||||||
private volatile ScheduledFuture<?> expirationFuture;
|
private volatile ScheduledFuture<?> expirationFuture;
|
||||||
|
|
||||||
DnsCacheEntry(String hostname, InetAddress address) {
|
public DnsCacheEntry(String hostname, InetAddress address) {
|
||||||
this.hostname = hostname;
|
this.hostname = checkNotNull(hostname, "hostname");
|
||||||
this.address = address;
|
this.address = checkNotNull(address, "address");
|
||||||
cause = null;
|
cause = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
DnsCacheEntry(String hostname, Throwable cause) {
|
public DnsCacheEntry(String hostname, Throwable cause) {
|
||||||
this.hostname = hostname;
|
this.hostname = checkNotNull(hostname, "hostname");
|
||||||
this.cause = cause;
|
this.cause = checkNotNull(cause, "cause");
|
||||||
address = null;
|
address = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
String hostname() {
|
public String hostname() {
|
||||||
return hostname;
|
return hostname;
|
||||||
}
|
}
|
||||||
|
|
||||||
InetAddress address() {
|
public InetAddress address() {
|
||||||
return address;
|
return address;
|
||||||
}
|
}
|
||||||
|
|
||||||
Throwable cause() {
|
public Throwable cause() {
|
||||||
return cause;
|
return cause;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -39,8 +39,6 @@ import io.netty.util.ReferenceCountUtil;
|
|||||||
import io.netty.util.concurrent.FastThreadLocal;
|
import io.netty.util.concurrent.FastThreadLocal;
|
||||||
import io.netty.util.concurrent.Future;
|
import io.netty.util.concurrent.Future;
|
||||||
import io.netty.util.concurrent.Promise;
|
import io.netty.util.concurrent.Promise;
|
||||||
import io.netty.util.internal.OneTimeTask;
|
|
||||||
import io.netty.util.internal.PlatformDependent;
|
|
||||||
import io.netty.util.internal.logging.InternalLogger;
|
import io.netty.util.internal.logging.InternalLogger;
|
||||||
import io.netty.util.internal.logging.InternalLoggerFactory;
|
import io.netty.util.internal.logging.InternalLoggerFactory;
|
||||||
|
|
||||||
@ -50,11 +48,7 @@ import java.net.InetSocketAddress;
|
|||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map.Entry;
|
|
||||||
import java.util.concurrent.ConcurrentMap;
|
|
||||||
import java.util.concurrent.TimeUnit;
|
|
||||||
|
|
||||||
import static io.netty.util.internal.ObjectUtil.*;
|
import static io.netty.util.internal.ObjectUtil.*;
|
||||||
|
|
||||||
@ -97,7 +91,7 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
/**
|
/**
|
||||||
* Cache for {@link #doResolve(String, Promise)} and {@link #doResolveAll(String, Promise)}.
|
* Cache for {@link #doResolve(String, Promise)} and {@link #doResolveAll(String, Promise)}.
|
||||||
*/
|
*/
|
||||||
private final ConcurrentMap<String, List<DnsCacheEntry>> resolveCache = PlatformDependent.newConcurrentHashMap();
|
private final DnsCache resolveCache;
|
||||||
|
|
||||||
private final FastThreadLocal<DnsServerAddressStream> nameServerAddrStream =
|
private final FastThreadLocal<DnsServerAddressStream> nameServerAddrStream =
|
||||||
new FastThreadLocal<DnsServerAddressStream>() {
|
new FastThreadLocal<DnsServerAddressStream>() {
|
||||||
@ -108,10 +102,6 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
};
|
};
|
||||||
|
|
||||||
private final long queryTimeoutMillis;
|
private final long queryTimeoutMillis;
|
||||||
// The default TTL values here respect the TTL returned by the DNS server and do not cache the negative response.
|
|
||||||
private final int minTtl;
|
|
||||||
private final int maxTtl;
|
|
||||||
private final int negativeTtl;
|
|
||||||
private final int maxQueriesPerResolve;
|
private final int maxQueriesPerResolve;
|
||||||
private final boolean traceEnabled;
|
private final boolean traceEnabled;
|
||||||
private final InternetProtocolFamily[] resolvedAddressTypes;
|
private final InternetProtocolFamily[] resolvedAddressTypes;
|
||||||
@ -129,9 +119,7 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
* @param nameServerAddresses the addresses of the DNS server. For each DNS query, a new stream is created from
|
* @param nameServerAddresses the addresses of the DNS server. For each DNS query, a new stream is created from
|
||||||
* this to determine which DNS server should be contacted for the next retry in case
|
* this to determine which DNS server should be contacted for the next retry in case
|
||||||
* of failure.
|
* of failure.
|
||||||
* @param minTtl the minimum TTL of cached DNS records
|
* @param resolveCache the DNS resolved entries cache
|
||||||
* @param maxTtl the maximum TTL of cached DNS records
|
|
||||||
* @param negativeTtl the TTL for failed cached queries
|
|
||||||
* @param queryTimeoutMillis timeout of each DNS query in millis
|
* @param queryTimeoutMillis timeout of each DNS query in millis
|
||||||
* @param resolvedAddressTypes list of the protocol families
|
* @param resolvedAddressTypes list of the protocol families
|
||||||
* @param recursionDesired if recursion desired flag must be set
|
* @param recursionDesired if recursion desired flag must be set
|
||||||
@ -146,9 +134,7 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
ChannelFactory<? extends DatagramChannel> channelFactory,
|
ChannelFactory<? extends DatagramChannel> channelFactory,
|
||||||
InetSocketAddress localAddress,
|
InetSocketAddress localAddress,
|
||||||
DnsServerAddresses nameServerAddresses,
|
DnsServerAddresses nameServerAddresses,
|
||||||
int minTtl,
|
DnsCache resolveCache,
|
||||||
int maxTtl,
|
|
||||||
int negativeTtl,
|
|
||||||
long queryTimeoutMillis,
|
long queryTimeoutMillis,
|
||||||
InternetProtocolFamily[] resolvedAddressTypes,
|
InternetProtocolFamily[] resolvedAddressTypes,
|
||||||
boolean recursionDesired,
|
boolean recursionDesired,
|
||||||
@ -162,13 +148,6 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
checkNotNull(channelFactory, "channelFactory");
|
checkNotNull(channelFactory, "channelFactory");
|
||||||
checkNotNull(localAddress, "localAddress");
|
checkNotNull(localAddress, "localAddress");
|
||||||
this.nameServerAddresses = checkNotNull(nameServerAddresses, "nameServerAddresses");
|
this.nameServerAddresses = checkNotNull(nameServerAddresses, "nameServerAddresses");
|
||||||
this.minTtl = checkPositiveOrZero(minTtl, "minTtl");
|
|
||||||
this.maxTtl = checkPositiveOrZero(maxTtl, "maxTtl");
|
|
||||||
if (minTtl > maxTtl) {
|
|
||||||
throw new IllegalArgumentException(
|
|
||||||
"minTtl: " + minTtl + ", maxTtl: " + maxTtl + " (expected: 0 <= minTtl <= maxTtl)");
|
|
||||||
}
|
|
||||||
this.negativeTtl = checkPositiveOrZero(negativeTtl, "negativeTtl");
|
|
||||||
this.queryTimeoutMillis = checkPositive(queryTimeoutMillis, "queryTimeoutMillis");
|
this.queryTimeoutMillis = checkPositive(queryTimeoutMillis, "queryTimeoutMillis");
|
||||||
this.resolvedAddressTypes = checkNonEmpty(resolvedAddressTypes, "resolvedAddressTypes");
|
this.resolvedAddressTypes = checkNonEmpty(resolvedAddressTypes, "resolvedAddressTypes");
|
||||||
this.recursionDesired = recursionDesired;
|
this.recursionDesired = recursionDesired;
|
||||||
@ -177,6 +156,7 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
this.maxPayloadSize = checkPositive(maxPayloadSize, "maxPayloadSize");
|
this.maxPayloadSize = checkPositive(maxPayloadSize, "maxPayloadSize");
|
||||||
this.optResourceEnabled = optResourceEnabled;
|
this.optResourceEnabled = optResourceEnabled;
|
||||||
this.hostsFileEntriesResolver = checkNotNull(hostsFileEntriesResolver, "hostsFileEntriesResolver");
|
this.hostsFileEntriesResolver = checkNotNull(hostsFileEntriesResolver, "hostsFileEntriesResolver");
|
||||||
|
this.resolveCache = resolveCache;
|
||||||
|
|
||||||
bindFuture = newChannel(channelFactory, localAddress);
|
bindFuture = newChannel(channelFactory, localAddress);
|
||||||
ch = (DatagramChannel) bindFuture.channel();
|
ch = (DatagramChannel) bindFuture.channel();
|
||||||
@ -201,7 +181,7 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
bindFuture.channel().closeFuture().addListener(new ChannelFutureListener() {
|
bindFuture.channel().closeFuture().addListener(new ChannelFutureListener() {
|
||||||
@Override
|
@Override
|
||||||
public void operationComplete(ChannelFuture future) throws Exception {
|
public void operationComplete(ChannelFuture future) throws Exception {
|
||||||
clearCache();
|
resolveCache.clear();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -209,29 +189,10 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the minimum TTL of the cached DNS resource records (in seconds).
|
* Returns the resolution cache.
|
||||||
*
|
|
||||||
* @see #maxTtl()
|
|
||||||
*/
|
*/
|
||||||
public int minTtl() {
|
public DnsCache resolveCache() {
|
||||||
return minTtl;
|
return resolveCache;
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the maximum TTL of the cached DNS resource records (in seconds).
|
|
||||||
*
|
|
||||||
* @see #minTtl()
|
|
||||||
*/
|
|
||||||
public int maxTtl() {
|
|
||||||
return maxTtl;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the TTL of the cache for the failed DNS queries (in seconds). The default value is {@code 0}, which
|
|
||||||
* disables the cache for negative results.
|
|
||||||
*/
|
|
||||||
public int negativeTtl() {
|
|
||||||
return negativeTtl;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -302,49 +263,6 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
return hostsFileEntriesResolver;
|
return hostsFileEntriesResolver;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Clears all the resolved addresses cached by this resolver.
|
|
||||||
*
|
|
||||||
* @return {@code this}
|
|
||||||
*
|
|
||||||
* @see #clearCache(String)
|
|
||||||
*/
|
|
||||||
public DnsNameResolver clearCache() {
|
|
||||||
for (Iterator<Entry<String, List<DnsCacheEntry>>> i = resolveCache.entrySet().iterator(); i.hasNext();) {
|
|
||||||
final Entry<String, List<DnsCacheEntry>> e = i.next();
|
|
||||||
i.remove();
|
|
||||||
cancelExpiration(e);
|
|
||||||
}
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Clears the resolved addresses of the specified host name from the cache of this resolver.
|
|
||||||
*
|
|
||||||
* @return {@code true} if and only if there was an entry for the specified host name in the cache and
|
|
||||||
* it has been removed by this method
|
|
||||||
*/
|
|
||||||
public boolean clearCache(String hostname) {
|
|
||||||
boolean removed = false;
|
|
||||||
for (Iterator<Entry<String, List<DnsCacheEntry>>> i = resolveCache.entrySet().iterator(); i.hasNext();) {
|
|
||||||
final Entry<String, List<DnsCacheEntry>> e = i.next();
|
|
||||||
if (e.getKey().equals(hostname)) {
|
|
||||||
i.remove();
|
|
||||||
cancelExpiration(e);
|
|
||||||
removed = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return removed;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void cancelExpiration(Entry<String, List<DnsCacheEntry>> e) {
|
|
||||||
final List<DnsCacheEntry> entries = e.getValue();
|
|
||||||
final int numEntries = entries.size();
|
|
||||||
for (int i = 0; i < numEntries; i++) {
|
|
||||||
entries.get(i).cancelExpiration();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Closes the internal datagram channel used for sending and receiving DNS messages, and clears all DNS resource
|
* Closes the internal datagram channel used for sending and receiving DNS messages, and clears all DNS resource
|
||||||
* records from the cache. Attempting to send a DNS query or to resolve a domain name will fail once this method
|
* records from the cache. Attempting to send a DNS query or to resolve a domain name will fail once this method
|
||||||
@ -366,6 +284,16 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void doResolve(String inetHost, Promise<InetAddress> promise) throws Exception {
|
protected void doResolve(String inetHost, Promise<InetAddress> promise) throws Exception {
|
||||||
|
doResolve(inetHost, promise, resolveCache);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Hook designed for extensibility so one can pass a different cache on each resolution attempt
|
||||||
|
* instead of using the global one.
|
||||||
|
*/
|
||||||
|
protected void doResolve(String inetHost,
|
||||||
|
Promise<InetAddress> promise,
|
||||||
|
DnsCache resolveCache) throws Exception {
|
||||||
final byte[] bytes = NetUtil.createByteArrayFromIpAddressString(inetHost);
|
final byte[] bytes = NetUtil.createByteArrayFromIpAddressString(inetHost);
|
||||||
if (bytes != null) {
|
if (bytes != null) {
|
||||||
// The inetHost is actually an ipaddress.
|
// The inetHost is actually an ipaddress.
|
||||||
@ -381,12 +309,14 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!doResolveCached(hostname, promise)) {
|
if (!doResolveCached(hostname, promise, resolveCache)) {
|
||||||
doResolveUncached(hostname, promise);
|
doResolveUncached(hostname, promise, resolveCache);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean doResolveCached(String hostname, Promise<InetAddress> promise) {
|
private boolean doResolveCached(String hostname,
|
||||||
|
Promise<InetAddress> promise,
|
||||||
|
DnsCache resolveCache) {
|
||||||
final List<DnsCacheEntry> cachedEntries = resolveCache.get(hostname);
|
final List<DnsCacheEntry> cachedEntries = resolveCache.get(hostname);
|
||||||
if (cachedEntries == null) {
|
if (cachedEntries == null) {
|
||||||
return false;
|
return false;
|
||||||
@ -433,9 +363,11 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void doResolveUncached(String hostname, Promise<InetAddress> promise) {
|
private void doResolveUncached(String hostname,
|
||||||
|
Promise<InetAddress> promise,
|
||||||
|
DnsCache resolveCache) {
|
||||||
final DnsNameResolverContext<InetAddress> ctx =
|
final DnsNameResolverContext<InetAddress> ctx =
|
||||||
new DnsNameResolverContext<InetAddress>(this, hostname, promise) {
|
new DnsNameResolverContext<InetAddress>(this, hostname, promise, resolveCache) {
|
||||||
@Override
|
@Override
|
||||||
protected boolean finishResolve(
|
protected boolean finishResolve(
|
||||||
Class<? extends InetAddress> addressType, List<DnsCacheEntry> resolvedEntries) {
|
Class<? extends InetAddress> addressType, List<DnsCacheEntry> resolvedEntries) {
|
||||||
@ -457,6 +389,16 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void doResolveAll(String inetHost, Promise<List<InetAddress>> promise) throws Exception {
|
protected void doResolveAll(String inetHost, Promise<List<InetAddress>> promise) throws Exception {
|
||||||
|
doResolveAll(inetHost, promise, resolveCache);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Hook designed for extensibility so one can pass a different cache on each resolution attempt
|
||||||
|
* instead of using the global one.
|
||||||
|
*/
|
||||||
|
protected void doResolveAll(String inetHost,
|
||||||
|
Promise<List<InetAddress>> promise,
|
||||||
|
DnsCache resolveCache) throws Exception {
|
||||||
|
|
||||||
final byte[] bytes = NetUtil.createByteArrayFromIpAddressString(inetHost);
|
final byte[] bytes = NetUtil.createByteArrayFromIpAddressString(inetHost);
|
||||||
if (bytes != null) {
|
if (bytes != null) {
|
||||||
@ -473,12 +415,14 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!doResolveAllCached(hostname, promise)) {
|
if (!doResolveAllCached(hostname, promise, resolveCache)) {
|
||||||
doResolveAllUncached(hostname, promise);
|
doResolveAllUncached(hostname, promise, resolveCache);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean doResolveAllCached(String hostname, Promise<List<InetAddress>> promise) {
|
private boolean doResolveAllCached(String hostname,
|
||||||
|
Promise<List<InetAddress>> promise,
|
||||||
|
DnsCache resolveCache) {
|
||||||
final List<DnsCacheEntry> cachedEntries = resolveCache.get(hostname);
|
final List<DnsCacheEntry> cachedEntries = resolveCache.get(hostname);
|
||||||
if (cachedEntries == null) {
|
if (cachedEntries == null) {
|
||||||
return false;
|
return false;
|
||||||
@ -518,9 +462,11 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void doResolveAllUncached(final String hostname, final Promise<List<InetAddress>> promise) {
|
private void doResolveAllUncached(final String hostname,
|
||||||
|
final Promise<List<InetAddress>> promise,
|
||||||
|
DnsCache resolveCache) {
|
||||||
final DnsNameResolverContext<List<InetAddress>> ctx =
|
final DnsNameResolverContext<List<InetAddress>> ctx =
|
||||||
new DnsNameResolverContext<List<InetAddress>>(this, hostname, promise) {
|
new DnsNameResolverContext<List<InetAddress>>(this, hostname, promise, resolveCache) {
|
||||||
@Override
|
@Override
|
||||||
protected boolean finishResolve(
|
protected boolean finishResolve(
|
||||||
Class<? extends InetAddress> addressType, List<DnsCacheEntry> resolvedEntries) {
|
Class<? extends InetAddress> addressType, List<DnsCacheEntry> resolvedEntries) {
|
||||||
@ -552,81 +498,6 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
return IDN.toASCII(inetHost);
|
return IDN.toASCII(inetHost);
|
||||||
}
|
}
|
||||||
|
|
||||||
void cache(String hostname, InetAddress address, long originalTtl) {
|
|
||||||
final int maxTtl = maxTtl();
|
|
||||||
if (maxTtl == 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
final int ttl = Math.max(minTtl(), (int) Math.min(maxTtl, originalTtl));
|
|
||||||
final List<DnsCacheEntry> entries = cachedEntries(hostname);
|
|
||||||
final DnsCacheEntry e = new DnsCacheEntry(hostname, address);
|
|
||||||
|
|
||||||
synchronized (entries) {
|
|
||||||
if (!entries.isEmpty()) {
|
|
||||||
final DnsCacheEntry firstEntry = entries.get(0);
|
|
||||||
if (firstEntry.cause() != null) {
|
|
||||||
assert entries.size() == 1;
|
|
||||||
firstEntry.cancelExpiration();
|
|
||||||
entries.clear();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
entries.add(e);
|
|
||||||
}
|
|
||||||
|
|
||||||
scheduleCacheExpiration(entries, e, ttl);
|
|
||||||
}
|
|
||||||
|
|
||||||
void cache(String hostname, Throwable cause) {
|
|
||||||
final int negativeTtl = negativeTtl();
|
|
||||||
if (negativeTtl == 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
final List<DnsCacheEntry> entries = cachedEntries(hostname);
|
|
||||||
final DnsCacheEntry e = new DnsCacheEntry(hostname, cause);
|
|
||||||
|
|
||||||
synchronized (entries) {
|
|
||||||
final int numEntries = entries.size();
|
|
||||||
for (int i = 0; i < numEntries; i ++) {
|
|
||||||
entries.get(i).cancelExpiration();
|
|
||||||
}
|
|
||||||
entries.clear();
|
|
||||||
entries.add(e);
|
|
||||||
}
|
|
||||||
|
|
||||||
scheduleCacheExpiration(entries, e, negativeTtl);
|
|
||||||
}
|
|
||||||
|
|
||||||
private List<DnsCacheEntry> cachedEntries(String hostname) {
|
|
||||||
List<DnsCacheEntry> oldEntries = resolveCache.get(hostname);
|
|
||||||
final List<DnsCacheEntry> entries;
|
|
||||||
if (oldEntries == null) {
|
|
||||||
List<DnsCacheEntry> newEntries = new ArrayList<DnsCacheEntry>();
|
|
||||||
oldEntries = resolveCache.putIfAbsent(hostname, newEntries);
|
|
||||||
entries = oldEntries != null? oldEntries : newEntries;
|
|
||||||
} else {
|
|
||||||
entries = oldEntries;
|
|
||||||
}
|
|
||||||
return entries;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void scheduleCacheExpiration(final List<DnsCacheEntry> entries, final DnsCacheEntry e, int ttl) {
|
|
||||||
e.scheduleExpiration(
|
|
||||||
ch.eventLoop(),
|
|
||||||
new OneTimeTask() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
synchronized (entries) {
|
|
||||||
entries.remove(e);
|
|
||||||
if (entries.isEmpty()) {
|
|
||||||
resolveCache.remove(e.hostname());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}, ttl, TimeUnit.SECONDS);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sends a DNS query with the specified question.
|
* Sends a DNS query with the specified question.
|
||||||
*/
|
*/
|
||||||
@ -700,9 +571,7 @@ public class DnsNameResolver extends InetNameResolver {
|
|||||||
|
|
||||||
final DnsQueryContext qCtx = queryContextManager.get(res.sender(), queryId);
|
final DnsQueryContext qCtx = queryContextManager.get(res.sender(), queryId);
|
||||||
if (qCtx == null) {
|
if (qCtx == null) {
|
||||||
if (logger.isWarnEnabled()) {
|
|
||||||
logger.warn("{} Received a DNS response with an unknown ID: {}", ch, queryId);
|
logger.warn("{} Received a DNS response with an unknown ID: {}", ch, queryId);
|
||||||
}
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -15,6 +15,8 @@
|
|||||||
*/
|
*/
|
||||||
package io.netty.resolver.dns;
|
package io.netty.resolver.dns;
|
||||||
|
|
||||||
|
import static io.netty.util.internal.ObjectUtil.intValue;
|
||||||
|
|
||||||
import io.netty.channel.ChannelFactory;
|
import io.netty.channel.ChannelFactory;
|
||||||
import io.netty.channel.EventLoop;
|
import io.netty.channel.EventLoop;
|
||||||
import io.netty.channel.ReflectiveChannelFactory;
|
import io.netty.channel.ReflectiveChannelFactory;
|
||||||
@ -37,9 +39,10 @@ public final class DnsNameResolverBuilder {
|
|||||||
private ChannelFactory<? extends DatagramChannel> channelFactory;
|
private ChannelFactory<? extends DatagramChannel> channelFactory;
|
||||||
private InetSocketAddress localAddress = DnsNameResolver.ANY_LOCAL_ADDR;
|
private InetSocketAddress localAddress = DnsNameResolver.ANY_LOCAL_ADDR;
|
||||||
private DnsServerAddresses nameServerAddresses;
|
private DnsServerAddresses nameServerAddresses;
|
||||||
private int minTtl;
|
private DnsCache resolveCache;
|
||||||
private int maxTtl = Integer.MAX_VALUE;
|
private Integer minTtl;
|
||||||
private int negativeTtl;
|
private Integer maxTtl;
|
||||||
|
private Integer negativeTtl;
|
||||||
private long queryTimeoutMillis = 5000;
|
private long queryTimeoutMillis = 5000;
|
||||||
private InternetProtocolFamily[] resolvedAddressTypes = DnsNameResolver.DEFAULT_RESOLVE_ADDRESS_TYPES;
|
private InternetProtocolFamily[] resolvedAddressTypes = DnsNameResolver.DEFAULT_RESOLVE_ADDRESS_TYPES;
|
||||||
private boolean recursionDesired = true;
|
private boolean recursionDesired = true;
|
||||||
@ -103,6 +106,17 @@ public final class DnsNameResolverBuilder {
|
|||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the cache for resolution results.
|
||||||
|
*
|
||||||
|
* @param resolveCache the DNS resolution results cache
|
||||||
|
* @return {@code this}
|
||||||
|
*/
|
||||||
|
public DnsNameResolverBuilder resolveCache(DnsCache resolveCache) {
|
||||||
|
this.resolveCache = resolveCache;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the minimum and maximum TTL of the cached DNS resource records (in seconds). If the TTL of the DNS
|
* Sets the minimum and maximum TTL of the cached DNS resource records (in seconds). If the TTL of the DNS
|
||||||
* resource record returned by the DNS server is less than the minimum TTL or greater than the maximum TTL,
|
* resource record returned by the DNS server is less than the minimum TTL or greater than the maximum TTL,
|
||||||
@ -291,14 +305,20 @@ public final class DnsNameResolverBuilder {
|
|||||||
* @return a {@link DnsNameResolver}
|
* @return a {@link DnsNameResolver}
|
||||||
*/
|
*/
|
||||||
public DnsNameResolver build() {
|
public DnsNameResolver build() {
|
||||||
|
|
||||||
|
if (resolveCache != null && (minTtl != null || maxTtl != null || negativeTtl != null)) {
|
||||||
|
throw new IllegalStateException("resolveCache and TTLs are mutually exclusive");
|
||||||
|
}
|
||||||
|
|
||||||
|
DnsCache cache = resolveCache != null ? resolveCache :
|
||||||
|
new DefaultDnsCache(intValue(minTtl, 0), intValue(maxTtl, Integer.MAX_VALUE), intValue(negativeTtl, 0));
|
||||||
|
|
||||||
return new DnsNameResolver(
|
return new DnsNameResolver(
|
||||||
eventLoop,
|
eventLoop,
|
||||||
channelFactory,
|
channelFactory,
|
||||||
localAddress,
|
localAddress,
|
||||||
nameServerAddresses,
|
nameServerAddresses,
|
||||||
minTtl,
|
cache,
|
||||||
maxTtl,
|
|
||||||
negativeTtl,
|
|
||||||
queryTimeoutMillis,
|
queryTimeoutMillis,
|
||||||
resolvedAddressTypes,
|
resolvedAddressTypes,
|
||||||
recursionDesired,
|
recursionDesired,
|
||||||
|
@ -70,6 +70,7 @@ abstract class DnsNameResolverContext<T> {
|
|||||||
private final DnsServerAddressStream nameServerAddrs;
|
private final DnsServerAddressStream nameServerAddrs;
|
||||||
private final Promise<T> promise;
|
private final Promise<T> promise;
|
||||||
private final String hostname;
|
private final String hostname;
|
||||||
|
private final DnsCache resolveCache;
|
||||||
private final boolean traceEnabled;
|
private final boolean traceEnabled;
|
||||||
private final int maxAllowedQueries;
|
private final int maxAllowedQueries;
|
||||||
private final InternetProtocolFamily[] resolveAddressTypes;
|
private final InternetProtocolFamily[] resolveAddressTypes;
|
||||||
@ -83,10 +84,14 @@ abstract class DnsNameResolverContext<T> {
|
|||||||
private int allowedQueries;
|
private int allowedQueries;
|
||||||
private boolean triedCNAME;
|
private boolean triedCNAME;
|
||||||
|
|
||||||
protected DnsNameResolverContext(DnsNameResolver parent, String hostname, Promise<T> promise) {
|
protected DnsNameResolverContext(DnsNameResolver parent,
|
||||||
|
String hostname,
|
||||||
|
Promise<T> promise,
|
||||||
|
DnsCache resolveCache) {
|
||||||
this.parent = parent;
|
this.parent = parent;
|
||||||
this.promise = promise;
|
this.promise = promise;
|
||||||
this.hostname = hostname;
|
this.hostname = hostname;
|
||||||
|
this.resolveCache = resolveCache;
|
||||||
|
|
||||||
nameServerAddrs = parent.nameServerAddresses.stream();
|
nameServerAddrs = parent.nameServerAddresses.stream();
|
||||||
maxAllowedQueries = parent.maxQueriesPerResolve();
|
maxAllowedQueries = parent.maxQueriesPerResolve();
|
||||||
@ -245,7 +250,7 @@ abstract class DnsNameResolverContext<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
final DnsCacheEntry e = new DnsCacheEntry(hostname, resolved);
|
final DnsCacheEntry e = new DnsCacheEntry(hostname, resolved);
|
||||||
parent.cache(hostname, resolved, r.timeToLive());
|
resolveCache.cache(hostname, resolved, r.timeToLive(), parent.ch.eventLoop());
|
||||||
resolvedEntries.add(e);
|
resolvedEntries.add(e);
|
||||||
found = true;
|
found = true;
|
||||||
|
|
||||||
@ -426,7 +431,7 @@ abstract class DnsNameResolverContext<T> {
|
|||||||
|
|
||||||
final UnknownHostException cause = new UnknownHostException(buf.toString());
|
final UnknownHostException cause = new UnknownHostException(buf.toString());
|
||||||
|
|
||||||
parent.cache(hostname, cause);
|
resolveCache.cache(hostname, cause, parent.ch.eventLoop());
|
||||||
promise.tryFailure(cause);
|
promise.tryFailure(cause);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -0,0 +1,63 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2016 The Netty Project
|
||||||
|
*
|
||||||
|
* The Netty Project licenses this file to you under the Apache License,
|
||||||
|
* version 2.0 (the "License"); you may not use this file except in compliance
|
||||||
|
* with the License. You may obtain a copy of the License at:
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing permissions and limitations
|
||||||
|
* under the License.
|
||||||
|
*/
|
||||||
|
package io.netty.resolver.dns;
|
||||||
|
|
||||||
|
import io.netty.channel.EventLoop;
|
||||||
|
|
||||||
|
import java.net.InetAddress;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A noop DNS cache that actually never caches anything.
|
||||||
|
*/
|
||||||
|
public final class NoopDnsCache implements DnsCache {
|
||||||
|
|
||||||
|
public static final NoopDnsCache INSTANCE = new NoopDnsCache();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Private singleton constructor.
|
||||||
|
*/
|
||||||
|
private NoopDnsCache() {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void clear() {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean clear(String hostname) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<DnsCacheEntry> get(String hostname) {
|
||||||
|
return Collections.emptyList();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void cache(String hostname, InetAddress address, long originalTtl, EventLoop loop) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void cache(String hostname, Throwable cause, EventLoop loop) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return NoopDnsCache.class.getSimpleName();
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user