YARN-7326. Add recursion support and configure RegistryDNS to lookup upstream. Contributed by Eric Yang

This commit is contained in:
Billie Rinaldi 2017-10-25 07:37:03 -07:00 committed by Jian He
parent 68acd88dcb
commit 501be9b4be
2 changed files with 148 additions and 8 deletions

View File

@ -37,8 +37,10 @@
import org.xbill.DNS.DNSSEC;
import org.xbill.DNS.DSRecord;
import org.xbill.DNS.ExtendedFlags;
import org.xbill.DNS.ExtendedResolver;
import org.xbill.DNS.Flags;
import org.xbill.DNS.Header;
import org.xbill.DNS.Lookup;
import org.xbill.DNS.Message;
import org.xbill.DNS.NSRecord;
import org.xbill.DNS.Name;
@ -49,9 +51,12 @@
import org.xbill.DNS.RRset;
import org.xbill.DNS.Rcode;
import org.xbill.DNS.Record;
import org.xbill.DNS.Resolver;
import org.xbill.DNS.ResolverConfig;
import org.xbill.DNS.SOARecord;
import org.xbill.DNS.Section;
import org.xbill.DNS.SetResponse;
import org.xbill.DNS.SimpleResolver;
import org.xbill.DNS.TSIG;
import org.xbill.DNS.TSIGRecord;
import org.xbill.DNS.TextParseException;
@ -66,8 +71,11 @@
import java.math.BigInteger;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.NetworkInterface;
import java.net.Socket;
import java.net.SocketAddress;
import java.net.SocketException;
import java.net.UnknownHostException;
import java.nio.ByteBuffer;
import java.nio.channels.DatagramChannel;
import java.nio.channels.ServerSocketChannel;
@ -78,10 +86,13 @@
import java.security.spec.InvalidKeySpecException;
import java.security.spec.RSAPrivateKeySpec;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Calendar;
import java.util.Collection;
import java.util.Date;
import java.util.Enumeration;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.concurrent.Callable;
@ -133,6 +144,16 @@ public class RegistryDNS extends AbstractService implements DNSOperations,
private boolean channelsInitialized = false;
/**
* Lock to update resolver only once per request.
*/
private final Object resolverUpdateLock = new Object();
/**
* Whether resolver update has been requested.
*/
private boolean resolverUpdateRequested = true;
/**
* Construct the service.
*
@ -171,6 +192,79 @@ public void initializeChannels(Configuration conf) throws Exception {
addNIOTCP(addr, port);
}
/**
* Initialize registryDNS to use /etc/resolv.conf values
* as default resolvers.
*/
private void updateDNSServer(Configuration conf) {
synchronized (resolverUpdateLock) {
if (!resolverUpdateRequested) {
return;
}
int port = conf.getInt(KEY_DNS_PORT, DEFAULT_DNS_PORT);
resolverUpdateRequested = false;
List<InetAddress> list = new ArrayList<InetAddress>();
try {
// If resolv.conf contains the server's own IP address,
// and RegistryDNS handles the lookup. Local IP address
// must be filter out from default resolvers to prevent
// self recursive loop.
if (port != 53) {
// When registryDNS is not running on default port,
// registryDNS can utilize local DNS server as upstream lookup.
throw new SocketException("Bypass filtering local DNS server.");
}
Enumeration<NetworkInterface> net =
NetworkInterface.getNetworkInterfaces();
while(net.hasMoreElements()) {
NetworkInterface n = (NetworkInterface) net.nextElement();
Enumeration<InetAddress> ee = n.getInetAddresses();
while (ee.hasMoreElements()) {
InetAddress i = (InetAddress) ee.nextElement();
list.add(i);
}
}
} catch (SocketException e) {
}
ResolverConfig.refresh();
ExtendedResolver resolver;
try {
resolver = new ExtendedResolver();
} catch (UnknownHostException e) {
LOG.error("Can not resolve DNS servers: ", e);
return;
}
for (Resolver check : resolver.getResolvers()) {
if (check instanceof SimpleResolver) {
InetAddress address = ((SimpleResolver) check).getAddress()
.getAddress();
if (list.contains(address)) {
resolver.deleteResolver(check);
continue;
} else {
check.setTimeout(30);
}
} else {
LOG.error("Not simple resolver!!!?" + check);
}
}
synchronized (Lookup.class) {
Lookup.setDefaultResolver(resolver);
Lookup.setDefaultSearchPath(ResolverConfig.getCurrentConfig()
.searchPath());
}
StringBuilder message = new StringBuilder();
message.append("DNS servers: ");
if (ResolverConfig.getCurrentConfig().servers() != null) {
for (String server : ResolverConfig.getCurrentConfig()
.servers()) {
message.append(server);
message.append(" ");
}
}
LOG.info(message.toString());
}
}
/**
* Initializes the registry.
*
@ -183,6 +277,7 @@ protected void serviceInit(Configuration conf) throws Exception {
// create the zone. for now create a "dummy" SOA record
try {
updateDNSServer(conf);
setDomainName(conf);
initializeZones(conf);
@ -916,9 +1011,7 @@ byte[] buildErrorMessage(Header header, int rcode, Record question) {
for (int i = 0; i < 4; i++) {
response.removeAllRecords(i);
}
if (rcode == Rcode.SERVFAIL) {
response.addRecord(question, Section.QUESTION);
}
response.addRecord(question, Section.QUESTION);
header.setRcode(rcode);
return response.toWire();
}
@ -975,6 +1068,7 @@ byte[] generateReply(Message query, Socket s)
response.getHeader().setFlag(Flags.QR);
if (query.getHeader().getFlag(Flags.RD)) {
response.getHeader().setFlag(Flags.RD);
response.getHeader().setFlag(Flags.RA);
}
response.addRecord(queryRecord, Section.QUESTION);
@ -992,10 +1086,10 @@ byte[] generateReply(Message query, Socket s)
LOG.debug("calling addAnswer");
byte rcode = addAnswer(response, name, type, dclass, 0, flags);
if (rcode != Rcode.NOERROR && rcode != Rcode.NXDOMAIN) {
return errorMessage(query, rcode);
if (rcode != Rcode.NOERROR) {
rcode = remoteLookup(response, name);
response.getHeader().setRcode(rcode);
}
addAdditional(response, flags);
if (queryOPT != null) {
@ -1008,6 +1102,45 @@ byte[] generateReply(Message query, Socket s)
return response.toWire(maxLength);
}
/**
* Lookup record from upstream DNS servers.
*/
private byte remoteLookup(Message response, Name name) {
// Forward lookup to primary DNS servers
Record[] answers = getRecords(name, Type.ANY);
try {
for (Record r : answers) {
if (r.getType() == Type.SOA) {
response.addRecord(r, Section.AUTHORITY);
} else {
response.addRecord(r, Section.ANSWER);
}
}
} catch (NullPointerException e) {
return Rcode.NXDOMAIN;
} catch (Throwable e) {
return Rcode.SERVFAIL;
}
return Rcode.NOERROR;
}
/**
* Requests records for the given resource name.
*
* @param name - query string
* @param type - type of DNS record to lookup
* @return DNS records
*/
protected Record[] getRecords(Name name, int type) {
try {
return new Lookup(name, type).run();
} catch (NullPointerException |
ExceptionInInitializerError e) {
LOG.error("Fail to lookup: " + name, e);
}
return null;
}
/**
* Create a query to forward to the primary DNS server (if configured).
* NOTE: Experimental
@ -1180,7 +1313,7 @@ byte addAnswer(Message response, Name name, int type, int dclass,
rcode = Rcode.NOTAUTH;
}
}
LOG.info("found record? {}", sr != null && sr.isSuccessful());
LOG.info("found local record? {}", sr != null && sr.isSuccessful());
if (sr != null) {
if (sr.isCNAME()) {
@ -1241,6 +1374,7 @@ byte addAnswer(Message response, Name name, int type, int dclass,
}
}
}
return rcode;
}

View File

@ -333,7 +333,7 @@ public void testMissingReverseLookup() throws Exception {
query.addRecord(optRecord, Section.ADDITIONAL);
byte[] responseBytes = getRegistryDNS().generateReply(query, null);
Message response = new Message(responseBytes);
assertEquals("No answer should be returned", Rcode.NOTAUTH,
assertEquals("Missing record should be: ", Rcode.NXDOMAIN,
response.getRcode());
}
@ -601,6 +601,12 @@ public void testSplitReverseZoneNames() throws Exception {
assertEquals(4, registryDNS.getZoneCount());
}
@Test
public void testExampleDotCom() throws Exception {
Name name = Name.fromString("example.com.");
Record[] records = getRegistryDNS().getRecords(name, Type.SOA);
assertNotNull("example.com exists:", records);
}
public RegistryDNS getRegistryDNS() {
return registryDNS;
}