Commit 48d78c27 authored by guus's avatar guus

OF-584: The ordered lists of hosts that is resolved through DNS SRV should not...

OF-584: The ordered lists of hosts that is resolved through DNS SRV should not only be prioritized, but also randomized based on weight value. As a result, connection attempts should be divided over all hosts that share the highest priority.

git-svn-id: http://svn.igniterealtime.org/svn/repos/openfire/trunk@13342 b35dd754-fafc-0310-a699-88a17e54d16e
parent 83bf58e2
......@@ -20,6 +20,7 @@
package org.jivesoftware.openfire.net;
import org.eclipse.jetty.util.MultiMap;
import org.jivesoftware.util.JiveGlobals;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
......@@ -32,14 +33,7 @@ import javax.naming.directory.DirContext;
import javax.naming.directory.InitialDirContext;
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Comparator;
import java.util.HashMap;
import java.util.Hashtable;
import java.util.List;
import java.util.Map;
import java.util.StringTokenizer;
import java.util.*;
/**
* Utilty class to perform DNS lookups for XMPP services.
......@@ -117,20 +111,19 @@ public class DNSUtil {
*/
public static List<HostAddress> resolveXMPPDomain(String domain, int defaultPort) {
// Check if there is an entry in the internal DNS for the specified domain
List<HostAddress> results = null;
List<HostAddress> results = new LinkedList<HostAddress>();
if (dnsOverride != null) {
HostAddress hostAddress = dnsOverride.get(domain);
if (hostAddress != null) {
results = new ArrayList<HostAddress>();
results.add(hostAddress);
return results;
}
}
// Attempt the SRV lookup.
results = srvLookup("_xmpp-server._tcp." + domain);
if (results == null || results.isEmpty()) {
results = srvLookup("_jabber._tcp." + domain);
results.addAll(srvLookup("_xmpp-server._tcp." + domain));
if (results.isEmpty()) {
results.addAll(srvLookup("_jabber._tcp." + domain));
}
// Use domain and default port as fallback.
......@@ -191,7 +184,7 @@ public class DNSUtil {
return answer;
}
private static List<HostAddress> srvLookup(String lookup) {
private static List<? extends HostAddress> srvLookup(String lookup) {
if (lookup == null) {
throw new NullPointerException("DNS lookup can't be null");
}
......@@ -203,14 +196,12 @@ public class DNSUtil {
logger.debug("No SRV record found for domain: " + lookup);
return new ArrayList<HostAddress>();
}
HostAddress[] hosts = new WeightedHostAddress[srvRecords.size()];
WeightedHostAddress[] hosts = new WeightedHostAddress[srvRecords.size()];
for (int i = 0; i < srvRecords.size(); i++) {
hosts[i] = new WeightedHostAddress(((String)srvRecords.get(i)).split(" "));
}
if (srvRecords.size() > 1) {
Arrays.sort(hosts, new SrvRecordWeightedPriorityComparator());
}
return Arrays.asList(hosts);
return prioritize(hosts);
}
catch (NameNotFoundException e) {
logger.debug("No SRV record found for: " + lookup, e);
......@@ -264,6 +255,67 @@ public class DNSUtil {
}
}
public static List<WeightedHostAddress> prioritize(WeightedHostAddress[] records) {
final List<WeightedHostAddress> result = new LinkedList<WeightedHostAddress>();
// sort by priority (ascending)
SortedMap<Integer, Set<WeightedHostAddress>> byPriority = new TreeMap<Integer, Set<WeightedHostAddress>>();
for(final WeightedHostAddress record : records) {
if (byPriority.containsKey(record.getPriority())) {
byPriority.get(record.getPriority()).add(record);
} else {
final Set<WeightedHostAddress> set = new HashSet<WeightedHostAddress>();
set.add(record);
byPriority.put(record.getPriority(), set);
}
}
// now, randomize each priority set by weight.
for(Map.Entry<Integer, Set<WeightedHostAddress>> weights : byPriority.entrySet()) {
List<WeightedHostAddress> zeroWeights = new LinkedList<WeightedHostAddress>();
int totalWeight = 0;
final Iterator<WeightedHostAddress> i = weights.getValue().iterator();
while (i.hasNext()) {
final WeightedHostAddress next = i.next();
if (next.weight == 0) {
// set aside, as these should be considered last according to the RFC.
zeroWeights.add(next);
i.remove();
continue;
}
totalWeight += next.getWeight();
}
int iterationWeight = totalWeight;
Iterator<WeightedHostAddress> iter = weights.getValue().iterator();
while (iter.hasNext()) {
int needle = new Random().nextInt(iterationWeight);
while (true) {
final WeightedHostAddress record = iter.next();
needle -= record.getWeight();
if (needle <= 0) {
result.add(record);
iter.remove();
iterationWeight -= record.getWeight();
break;
}
}
iter = weights.getValue().iterator();
}
// finally, append the hosts with zero priority (shuffled)
Collections.shuffle(zeroWeights);
for(WeightedHostAddress zero : zeroWeights) {
result.add(zero);
}
}
return result;
}
/**
* The representation of weighted address.
*/
......@@ -279,7 +331,7 @@ public class DNSUtil {
priority = Integer.parseInt(srvRecordEntries[srvRecordEntries.length-4]);
}
private WeightedHostAddress(String host, int port, int priority, int weight) {
WeightedHostAddress(String host, int port, int priority, int weight) {
super(host, port);
this.priority = priority;
this.weight = weight;
......@@ -303,24 +355,4 @@ public class DNSUtil {
return weight;
}
}
/**
* A comparator for sorting multiple weighted host addresses according to RFC 2782.
*/
public static class SrvRecordWeightedPriorityComparator implements Comparator<HostAddress>, Serializable {
private static final long serialVersionUID = -9207293572898848260L;
public int compare(HostAddress o1, HostAddress o2) {
if (o1 instanceof WeightedHostAddress && o2 instanceof WeightedHostAddress) {
WeightedHostAddress srv1 = (WeightedHostAddress) o1;
WeightedHostAddress srv2 = (WeightedHostAddress) o2;
// 16 bit unsigned priority is more important as the 16 bit weight
return ((srv1.priority << 15) - (srv2.priority << 15)) + (srv2.weight - srv1.weight);
}
else {
// This shouldn't happen but if we don't have priorities we sort the addresses
return o1.toString().compareTo(o2.toString());
}
}
}
}
\ No newline at end of file
package org.jivesoftware.openfire.net;
import org.jivesoftware.openfire.net.DNSUtil;
import org.junit.Assert;
import org.junit.BeforeClass;
import org.junit.Test;
import java.util.List;
/**
* Unit tests for {@link org.jivesoftware.openfire.net.DNSUtil}.
*
* @author Guus der Kinderen, guus.der.kinderen@gmail.com
*/
public class DNSUtilTest {
//@Test
public void testJabberDotOrg() throws Exception {
for (int i=0; i<=10; i++) {
final List<DNSUtil.HostAddress> list = DNSUtil.resolveXMPPDomain("jabber.org", 5222);
for(DNSUtil.HostAddress address : list) {
System.out.println("Address: " + address.toString());
}
System.out.println("");
}
}
/**
* Runs {@link DNSUtil#prioritize(org.jivesoftware.openfire.net.DNSUtil.WeightedHostAddress[])} on a copy of the
* DNS SRV xmpp-server records for jabber.org (as they were last 2012).
*/
@Test
public void testJabberDotOrgMock() throws Exception {
// setup
final DNSUtil.WeightedHostAddress fallback = new DNSUtil.WeightedHostAddress("fallback.jabber.org", 5269, 31, 31);
final DNSUtil.WeightedHostAddress hermes6 = new DNSUtil.WeightedHostAddress("hermes6.jabber.org", 5269, 30, 30);
final DNSUtil.WeightedHostAddress hermes = new DNSUtil.WeightedHostAddress("hermes.jabber.org", 5269, 30, 30);
// do magic
final List<DNSUtil.WeightedHostAddress> result = DNSUtil.prioritize(new DNSUtil.WeightedHostAddress[]{fallback, hermes6, hermes});
// verify
Assert.assertEquals("There were three records in the input, the output should have contained the same amount.", 3, result.size());
Assert.assertTrue("The 'hermes' host should have been included somewhere in the output." , result.contains(hermes));
Assert.assertTrue("The 'hermes6' host should have been included somewhere in the output." , result.contains(hermes6));
Assert.assertTrue("The 'fallback' host should bhave been included somewhere in the output.", result.contains(fallback));
Assert.assertEquals("The 'fallback' host should have been the last record in the result." , fallback, result.get(2));
}
/**
* A basic check that verifies that when one hosts exists, it gets returned in the output.
*/
@Test
public void testOneHost() throws Exception {
// setup
final DNSUtil.WeightedHostAddress host = new DNSUtil.WeightedHostAddress("host", 5222, 1, 1);
// do magic
final List<DNSUtil.WeightedHostAddress> result = DNSUtil.prioritize(new DNSUtil.WeightedHostAddress[]{host});
// verify
Assert.assertEquals(1, result.size());
Assert.assertEquals(host, result.get(0));
}
/**
* A check equal to {@link #testOneHost()}, but using (the edge-case) priority value of zero.
*/
@Test
public void testOneHostZeroPiority() throws Exception {
// setup
final DNSUtil.WeightedHostAddress host = new DNSUtil.WeightedHostAddress("host", 5222, 0, 1);
// do magic
final List<DNSUtil.WeightedHostAddress> result = DNSUtil.prioritize(new DNSUtil.WeightedHostAddress[]{host});
// verify
Assert.assertEquals(1, result.size());
Assert.assertEquals(host, result.get(0));
}
/**
* A check equal to {@link #testOneHost()}, but using (the edge-case) weight value of zero.
*/
@Test
public void testOneHostZeroWeight() throws Exception {
// setup
final DNSUtil.WeightedHostAddress host = new DNSUtil.WeightedHostAddress("host", 5222, 1, 0);
// do magic
final List<DNSUtil.WeightedHostAddress> result = DNSUtil.prioritize(new DNSUtil.WeightedHostAddress[]{host});
// verify
Assert.assertEquals(1, result.size());
Assert.assertEquals(host, result.get(0));
}
/**
* Verifies that when a couple of records exist that all have a particular priority, those records are all included
* in the result, ordered (ascending) by their priority.
* @throws Exception
*/
@Test
public void testDifferentPriorities() throws Exception {
// setup
final DNSUtil.WeightedHostAddress hostA = new DNSUtil.WeightedHostAddress("hostA", 5222, 1, 1);
final DNSUtil.WeightedHostAddress hostB = new DNSUtil.WeightedHostAddress("hostB", 5222, 3, 1);
final DNSUtil.WeightedHostAddress hostC = new DNSUtil.WeightedHostAddress("hostC", 5222, 2, 1);
// do magic
final List<DNSUtil.WeightedHostAddress> result = DNSUtil.prioritize(new DNSUtil.WeightedHostAddress[]{hostA, hostB, hostC});
// verify
Assert.assertEquals(3, result.size());
Assert.assertEquals(hostA, result.get(0));
Assert.assertEquals(hostC, result.get(1));
Assert.assertEquals(hostB, result.get(2));
}
/**
* A test equal to {@link #testDifferentPriorities()}, but with one of the priorities set to zero.
*/
@Test
public void testZeroPriority() throws Exception {
// setup
final DNSUtil.WeightedHostAddress hostA = new DNSUtil.WeightedHostAddress("hostA", 5222, 0, 1);
final DNSUtil.WeightedHostAddress hostB = new DNSUtil.WeightedHostAddress("hostB", 5222, 2, 1);
final DNSUtil.WeightedHostAddress hostC = new DNSUtil.WeightedHostAddress("hostC", 5222, 1, 1);
// do magic
final List<DNSUtil.WeightedHostAddress> result = DNSUtil.prioritize(new DNSUtil.WeightedHostAddress[]{hostA, hostB, hostC});
// verify
Assert.assertEquals(3, result.size());
Assert.assertEquals(hostA, result.get(0));
Assert.assertEquals(hostC, result.get(1));
Assert.assertEquals(hostB, result.get(2));
}
/**
* A test that verifies that hosts with equal weight are alternately first in the resulting list.
*
* The test that is done here re-executes until each of the input records was included in the output as the first
* record. This indicates that the returning list is at the very least not always ordered in the exact same way.
*/
@Test
public void testSameWeights() throws Exception {
// setup
final DNSUtil.WeightedHostAddress hostA = new DNSUtil.WeightedHostAddress("hostA", 5222, 1, 10);
final DNSUtil.WeightedHostAddress hostB = new DNSUtil.WeightedHostAddress("hostB", 5222, 1, 10);
final DNSUtil.WeightedHostAddress[] hosts = new DNSUtil.WeightedHostAddress[] { hostA, hostB };
// do magic
boolean hostAWasFirst = false;
boolean hostBWasFirst = false;
final int maxTries = Integer.MAX_VALUE;
for (int i=0; i<maxTries; i++) {
final List<DNSUtil.WeightedHostAddress> result = DNSUtil.prioritize(hosts);
if (hostA.equals(result.get(0))) {
hostAWasFirst = true;
}
if (hostB.equals(result.get(0))) {
hostBWasFirst = true;
}
if (hostAWasFirst && hostBWasFirst) {
break;
}
if (i%1000000==0 && i>0) {
System.err.println("The last " + i + " iterations of this test all had the same result, which is very unlikely to occur (there should be an even distribution between two possible outcomes). We'll iterate up to "+ maxTries +" times, but you might want to abort the unit test at this point...");
}
}
// verify
Assert.assertTrue(hostAWasFirst);
Assert.assertTrue(hostBWasFirst);
}
/**
* A test equal to {@link #testSameWeights()}, but using records with a weight of zero.
*/
@Test
public void testZeroWeights() throws Exception {
// setup
final DNSUtil.WeightedHostAddress hostA = new DNSUtil.WeightedHostAddress("hostA", 5222, 1, 0);
final DNSUtil.WeightedHostAddress hostB = new DNSUtil.WeightedHostAddress("hostB", 5222, 1, 0);
final DNSUtil.WeightedHostAddress[] hosts = new DNSUtil.WeightedHostAddress[] { hostA, hostB };
// do magic
boolean hostAWasFirst = false;
boolean hostBWasFirst = false;
final int maxTries = Integer.MAX_VALUE;
for (int i=0; i<maxTries; i++) {
final List<DNSUtil.WeightedHostAddress> result = DNSUtil.prioritize(hosts);
if (hostA.equals(result.get(0))) {
hostAWasFirst = true;
}
if (hostB.equals(result.get(0))) {
hostBWasFirst = true;
}
if (hostAWasFirst && hostBWasFirst) {
break;
}
if (i%1000000==0 && i>0) {
System.err.println("The last " + i + " iterations of this test all had the same result, which is very unlikely to occur (there should be an even distribution between two possible outcomes). We'll iterate up to "+ maxTries +" times, but you might want to abort the unit test at this point...");
}
}
// verify
Assert.assertTrue(hostAWasFirst);
Assert.assertTrue(hostBWasFirst);
}
}
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment