diff --git a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/net/DFSNetworkTopology.java b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/net/DFSNetworkTopology.java index aea1ff6cd97..886411a62d3 100644 --- a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/net/DFSNetworkTopology.java +++ b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/net/DFSNetworkTopology.java @@ -212,8 +212,7 @@ public class DFSNetworkTopology extends NetworkTopology { } if (excludedNodes != null) { for (Node excludedNode : excludedNodes) { - if (excludeRoot != null - && excludedNode.getNetworkLocation().startsWith(excludedScope)) { + if (excludeRoot != null && isNodeInScope(excludedNode, excludedScope)) { continue; } if (excludedNode instanceof DatanodeDescriptor) { @@ -259,6 +258,14 @@ public class DFSNetworkTopology extends NetworkTopology { return chosen; } + private boolean isNodeInScope(Node node, String scope) { + if (!scope.endsWith("/")) { + scope += "/"; + } + String nodeLocation = node.getNetworkLocation() + "/"; + return nodeLocation.startsWith(scope); + } + /** * Choose a random node that has the required storage type, under the given * root, with an excluded subtree root (could also just be a leaf node). diff --git a/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/net/TestDFSNetworkTopology.java b/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/net/TestDFSNetworkTopology.java index fdb41a206cd..380466e4f64 100644 --- a/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/net/TestDFSNetworkTopology.java +++ b/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/net/TestDFSNetworkTopology.java @@ -581,4 +581,23 @@ public class TestDFSNetworkTopology { assertTrue(dd.getHostName().equals("host7")); } } + + @Test + public void testChooseRandomWithStorageTypeNoAvlblNode() { + DFSNetworkTopology dfsCluster = + DFSNetworkTopology.getInstance(new Configuration()); + final String[] racks = {"/default/rack1", "/default/rack10"}; + final String[] hosts = {"host1", "host2"}; + final StorageType[] types = {StorageType.DISK, StorageType.DISK}; + final DatanodeStorageInfo[] storages = + DFSTestUtil.createDatanodeStorageInfos(2, racks, hosts, types); + DatanodeDescriptor[] dns = DFSTestUtil.toDatanodeDescriptor(storages); + dfsCluster.add(dns[0]); + dfsCluster.add(dns[1]); + HashSet excluded = new HashSet<>(); + excluded.add(dns[1]); + Node n = dfsCluster.chooseRandomWithStorageType("/default", + "/default/rack1", excluded, StorageType.DISK); + assertNull("No node should have been selected.", n); + } }