YARN-7326. Add recursion support and configure RegistryDNS to lookup upstream. Contributed by Eric Yang
This commit is contained in:
parent
68acd88dcb
commit
501be9b4be
@ -37,8 +37,10 @@
|
|||||||
import org.xbill.DNS.DNSSEC;
|
import org.xbill.DNS.DNSSEC;
|
||||||
import org.xbill.DNS.DSRecord;
|
import org.xbill.DNS.DSRecord;
|
||||||
import org.xbill.DNS.ExtendedFlags;
|
import org.xbill.DNS.ExtendedFlags;
|
||||||
|
import org.xbill.DNS.ExtendedResolver;
|
||||||
import org.xbill.DNS.Flags;
|
import org.xbill.DNS.Flags;
|
||||||
import org.xbill.DNS.Header;
|
import org.xbill.DNS.Header;
|
||||||
|
import org.xbill.DNS.Lookup;
|
||||||
import org.xbill.DNS.Message;
|
import org.xbill.DNS.Message;
|
||||||
import org.xbill.DNS.NSRecord;
|
import org.xbill.DNS.NSRecord;
|
||||||
import org.xbill.DNS.Name;
|
import org.xbill.DNS.Name;
|
||||||
@ -49,9 +51,12 @@
|
|||||||
import org.xbill.DNS.RRset;
|
import org.xbill.DNS.RRset;
|
||||||
import org.xbill.DNS.Rcode;
|
import org.xbill.DNS.Rcode;
|
||||||
import org.xbill.DNS.Record;
|
import org.xbill.DNS.Record;
|
||||||
|
import org.xbill.DNS.Resolver;
|
||||||
|
import org.xbill.DNS.ResolverConfig;
|
||||||
import org.xbill.DNS.SOARecord;
|
import org.xbill.DNS.SOARecord;
|
||||||
import org.xbill.DNS.Section;
|
import org.xbill.DNS.Section;
|
||||||
import org.xbill.DNS.SetResponse;
|
import org.xbill.DNS.SetResponse;
|
||||||
|
import org.xbill.DNS.SimpleResolver;
|
||||||
import org.xbill.DNS.TSIG;
|
import org.xbill.DNS.TSIG;
|
||||||
import org.xbill.DNS.TSIGRecord;
|
import org.xbill.DNS.TSIGRecord;
|
||||||
import org.xbill.DNS.TextParseException;
|
import org.xbill.DNS.TextParseException;
|
||||||
@ -66,8 +71,11 @@
|
|||||||
import java.math.BigInteger;
|
import java.math.BigInteger;
|
||||||
import java.net.InetAddress;
|
import java.net.InetAddress;
|
||||||
import java.net.InetSocketAddress;
|
import java.net.InetSocketAddress;
|
||||||
|
import java.net.NetworkInterface;
|
||||||
import java.net.Socket;
|
import java.net.Socket;
|
||||||
import java.net.SocketAddress;
|
import java.net.SocketAddress;
|
||||||
|
import java.net.SocketException;
|
||||||
|
import java.net.UnknownHostException;
|
||||||
import java.nio.ByteBuffer;
|
import java.nio.ByteBuffer;
|
||||||
import java.nio.channels.DatagramChannel;
|
import java.nio.channels.DatagramChannel;
|
||||||
import java.nio.channels.ServerSocketChannel;
|
import java.nio.channels.ServerSocketChannel;
|
||||||
@ -78,10 +86,13 @@
|
|||||||
import java.security.spec.InvalidKeySpecException;
|
import java.security.spec.InvalidKeySpecException;
|
||||||
import java.security.spec.RSAPrivateKeySpec;
|
import java.security.spec.RSAPrivateKeySpec;
|
||||||
import java.text.SimpleDateFormat;
|
import java.text.SimpleDateFormat;
|
||||||
|
import java.util.ArrayList;
|
||||||
import java.util.Calendar;
|
import java.util.Calendar;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
|
import java.util.Enumeration;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
import java.util.concurrent.Callable;
|
import java.util.concurrent.Callable;
|
||||||
@ -133,6 +144,16 @@ public class RegistryDNS extends AbstractService implements DNSOperations,
|
|||||||
|
|
||||||
private boolean channelsInitialized = false;
|
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.
|
* Construct the service.
|
||||||
*
|
*
|
||||||
@ -171,6 +192,79 @@ public void initializeChannels(Configuration conf) throws Exception {
|
|||||||
addNIOTCP(addr, port);
|
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.
|
* Initializes the registry.
|
||||||
*
|
*
|
||||||
@ -183,6 +277,7 @@ protected void serviceInit(Configuration conf) throws Exception {
|
|||||||
|
|
||||||
// create the zone. for now create a "dummy" SOA record
|
// create the zone. for now create a "dummy" SOA record
|
||||||
try {
|
try {
|
||||||
|
updateDNSServer(conf);
|
||||||
setDomainName(conf);
|
setDomainName(conf);
|
||||||
|
|
||||||
initializeZones(conf);
|
initializeZones(conf);
|
||||||
@ -916,9 +1011,7 @@ byte[] buildErrorMessage(Header header, int rcode, Record question) {
|
|||||||
for (int i = 0; i < 4; i++) {
|
for (int i = 0; i < 4; i++) {
|
||||||
response.removeAllRecords(i);
|
response.removeAllRecords(i);
|
||||||
}
|
}
|
||||||
if (rcode == Rcode.SERVFAIL) {
|
|
||||||
response.addRecord(question, Section.QUESTION);
|
response.addRecord(question, Section.QUESTION);
|
||||||
}
|
|
||||||
header.setRcode(rcode);
|
header.setRcode(rcode);
|
||||||
return response.toWire();
|
return response.toWire();
|
||||||
}
|
}
|
||||||
@ -975,6 +1068,7 @@ byte[] generateReply(Message query, Socket s)
|
|||||||
response.getHeader().setFlag(Flags.QR);
|
response.getHeader().setFlag(Flags.QR);
|
||||||
if (query.getHeader().getFlag(Flags.RD)) {
|
if (query.getHeader().getFlag(Flags.RD)) {
|
||||||
response.getHeader().setFlag(Flags.RD);
|
response.getHeader().setFlag(Flags.RD);
|
||||||
|
response.getHeader().setFlag(Flags.RA);
|
||||||
}
|
}
|
||||||
response.addRecord(queryRecord, Section.QUESTION);
|
response.addRecord(queryRecord, Section.QUESTION);
|
||||||
|
|
||||||
@ -992,10 +1086,10 @@ byte[] generateReply(Message query, Socket s)
|
|||||||
|
|
||||||
LOG.debug("calling addAnswer");
|
LOG.debug("calling addAnswer");
|
||||||
byte rcode = addAnswer(response, name, type, dclass, 0, flags);
|
byte rcode = addAnswer(response, name, type, dclass, 0, flags);
|
||||||
if (rcode != Rcode.NOERROR && rcode != Rcode.NXDOMAIN) {
|
if (rcode != Rcode.NOERROR) {
|
||||||
return errorMessage(query, rcode);
|
rcode = remoteLookup(response, name);
|
||||||
|
response.getHeader().setRcode(rcode);
|
||||||
}
|
}
|
||||||
|
|
||||||
addAdditional(response, flags);
|
addAdditional(response, flags);
|
||||||
|
|
||||||
if (queryOPT != null) {
|
if (queryOPT != null) {
|
||||||
@ -1008,6 +1102,45 @@ byte[] generateReply(Message query, Socket s)
|
|||||||
return response.toWire(maxLength);
|
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).
|
* Create a query to forward to the primary DNS server (if configured).
|
||||||
* NOTE: Experimental
|
* NOTE: Experimental
|
||||||
@ -1180,7 +1313,7 @@ byte addAnswer(Message response, Name name, int type, int dclass,
|
|||||||
rcode = Rcode.NOTAUTH;
|
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 != null) {
|
||||||
if (sr.isCNAME()) {
|
if (sr.isCNAME()) {
|
||||||
@ -1241,6 +1374,7 @@ byte addAnswer(Message response, Name name, int type, int dclass,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return rcode;
|
return rcode;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -333,7 +333,7 @@ public void testMissingReverseLookup() throws Exception {
|
|||||||
query.addRecord(optRecord, Section.ADDITIONAL);
|
query.addRecord(optRecord, Section.ADDITIONAL);
|
||||||
byte[] responseBytes = getRegistryDNS().generateReply(query, null);
|
byte[] responseBytes = getRegistryDNS().generateReply(query, null);
|
||||||
Message response = new Message(responseBytes);
|
Message response = new Message(responseBytes);
|
||||||
assertEquals("No answer should be returned", Rcode.NOTAUTH,
|
assertEquals("Missing record should be: ", Rcode.NXDOMAIN,
|
||||||
response.getRcode());
|
response.getRcode());
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -601,6 +601,12 @@ public void testSplitReverseZoneNames() throws Exception {
|
|||||||
assertEquals(4, registryDNS.getZoneCount());
|
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() {
|
public RegistryDNS getRegistryDNS() {
|
||||||
return registryDNS;
|
return registryDNS;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user