diff --git a/hadoop-hdfs-project/hadoop-hdfs/CHANGES.txt b/hadoop-hdfs-project/hadoop-hdfs/CHANGES.txt index daf5822964d..a38d3b208a5 100644 --- a/hadoop-hdfs-project/hadoop-hdfs/CHANGES.txt +++ b/hadoop-hdfs-project/hadoop-hdfs/CHANGES.txt @@ -12,6 +12,9 @@ Release 2.7.4 - UNRELEASED BUG FIXES + HDFS-9601. NNThroughputBenchmark.BlockReportStats should handle + NotReplicatedYetException on adding block. (iwasakims) + Release 2.7.3 - UNRELEASED INCOMPATIBLE CHANGES diff --git a/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/server/namenode/NNThroughputBenchmark.java b/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/server/namenode/NNThroughputBenchmark.java index 9e24f7277ec..79cdb31a5f4 100644 --- a/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/server/namenode/NNThroughputBenchmark.java +++ b/hadoop-hdfs-project/hadoop-hdfs/src/test/java/org/apache/hadoop/hdfs/server/namenode/NNThroughputBenchmark.java @@ -60,6 +60,7 @@ import org.apache.hadoop.hdfs.server.protocol.StorageBlockReport; import org.apache.hadoop.hdfs.server.protocol.StorageReceivedDeletedBlocks; import org.apache.hadoop.hdfs.server.protocol.StorageReport; import org.apache.hadoop.io.EnumSetWritable; +import org.apache.hadoop.ipc.RemoteException; import org.apache.hadoop.net.DNS; import org.apache.hadoop.net.NetworkTopology; import org.apache.hadoop.security.Groups; @@ -1153,7 +1154,7 @@ public class NNThroughputBenchmark implements Tool { throws IOException { ExtendedBlock prevBlock = null; for(int jdx = 0; jdx < blocksPerFile; jdx++) { - LocatedBlock loc = nameNodeProto.addBlock(fileName, clientName, + LocatedBlock loc = addBlock(fileName, clientName, prevBlock, null, INodeId.GRANDFATHER_INODE_ID, null); prevBlock = loc.getBlock(); for(DatanodeInfo dnInfo : loc.getLocations()) { @@ -1167,10 +1168,41 @@ public class NNThroughputBenchmark implements Tool { nameNodeProto.blockReceivedAndDeleted(datanodes[dnIdx].dnRegistration, loc .getBlock().getBlockPoolId(), report); } + // IBRs are asynchronously processed by NameNode. The next + // ClientProtocol#addBlock() may throw NotReplicatedYetException. } return prevBlock; } + /** + * Retry ClientProtocol.addBlock() if it throws NotReplicatedYetException. + * Because addBlock() also commits the previous block, + * it fails if enough IBRs are not processed by NameNode. + */ + private LocatedBlock addBlock(String src, String clientName, + ExtendedBlock previous, DatanodeInfo[] excludeNodes, long fileId, + String[] favoredNodes) throws IOException { + for (int i = 0; i < 30; i++) { + try { + return nameNodeProto.addBlock(src, clientName, + previous, excludeNodes, fileId, favoredNodes); + } catch (NotReplicatedYetException|RemoteException e) { + if (e instanceof RemoteException) { + String className = ((RemoteException) e).getClassName(); + if (!className.equals(NotReplicatedYetException.class.getName())) { + throw e; + } + } + try { + Thread.sleep(100); + } catch (InterruptedException ie) { + LOG.warn("interrupted while retrying addBlock.", ie); + } + } + } + throw new IOException("failed to add block."); + } + /** * Does not require the argument */