From 41e18feda3f5ff924c87c4bed5b5cbbaecb19ae1 Mon Sep 17 00:00:00 2001 From: Surendra Singh Lilhore Date: Wed, 20 Feb 2019 16:56:10 +0530 Subject: [PATCH] HDFS-14235. Handle ArrayIndexOutOfBoundsException in DataNodeDiskMetrics#slowDiskDetectionDaemon. Contributed by Ranith Sardar. --- .../datanode/metrics/DataNodeDiskMetrics.java | 76 ++++++++++--------- 1 file changed, 42 insertions(+), 34 deletions(-) diff --git a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/datanode/metrics/DataNodeDiskMetrics.java b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/datanode/metrics/DataNodeDiskMetrics.java index f2954e824dc..a8a6c85762d 100644 --- a/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/datanode/metrics/DataNodeDiskMetrics.java +++ b/hadoop-hdfs-project/hadoop-hdfs/src/main/java/org/apache/hadoop/hdfs/server/datanode/metrics/DataNodeDiskMetrics.java @@ -57,6 +57,10 @@ public class DataNodeDiskMetrics { private volatile Map> diskOutliersStats = Maps.newHashMap(); + // Adding for test purpose. When addSlowDiskForTesting() called from test + // code, status should not be overridden by daemon thread. + private boolean overrideStatus = true; + public DataNodeDiskMetrics(DataNode dn, long diskOutlierDetectionIntervalMs) { this.dn = dn; this.detectionInterval = diskOutlierDetectionIntervalMs; @@ -71,41 +75,43 @@ public class DataNodeDiskMetrics { @Override public void run() { while (shouldRun) { - Map metadataOpStats = Maps.newHashMap(); - Map readIoStats = Maps.newHashMap(); - Map writeIoStats = Maps.newHashMap(); - FsDatasetSpi.FsVolumeReferences fsVolumeReferences = null; - try { - fsVolumeReferences = dn.getFSDataset().getFsVolumeReferences(); - Iterator volumeIterator = fsVolumeReferences - .iterator(); - while (volumeIterator.hasNext()) { - FsVolumeSpi volume = volumeIterator.next(); - DataNodeVolumeMetrics metrics = volumeIterator.next().getMetrics(); - String volumeName = volume.getBaseURI().getPath(); + if (dn.getFSDataset() != null) { + Map metadataOpStats = Maps.newHashMap(); + Map readIoStats = Maps.newHashMap(); + Map writeIoStats = Maps.newHashMap(); + FsDatasetSpi.FsVolumeReferences fsVolumeReferences = null; + try { + fsVolumeReferences = dn.getFSDataset().getFsVolumeReferences(); + Iterator volumeIterator = fsVolumeReferences + .iterator(); + while (volumeIterator.hasNext()) { + FsVolumeSpi volume = volumeIterator.next(); + DataNodeVolumeMetrics metrics = volume.getMetrics(); + String volumeName = volume.getBaseURI().getPath(); - metadataOpStats.put(volumeName, - metrics.getMetadataOperationMean()); - readIoStats.put(volumeName, metrics.getReadIoMean()); - writeIoStats.put(volumeName, metrics.getWriteIoMean()); - } - } finally { - if (fsVolumeReferences != null) { - try { - fsVolumeReferences.close(); - } catch (IOException e) { - LOG.error("Error in releasing FS Volume references", e); + metadataOpStats.put(volumeName, + metrics.getMetadataOperationMean()); + readIoStats.put(volumeName, metrics.getReadIoMean()); + writeIoStats.put(volumeName, metrics.getWriteIoMean()); + } + } finally { + if (fsVolumeReferences != null) { + try { + fsVolumeReferences.close(); + } catch (IOException e) { + LOG.error("Error in releasing FS Volume references", e); + } } } - } - if (metadataOpStats.isEmpty() && readIoStats.isEmpty() && - writeIoStats.isEmpty()) { - LOG.debug("No disk stats available for detecting outliers."); - return; - } + if (metadataOpStats.isEmpty() && readIoStats.isEmpty() + && writeIoStats.isEmpty()) { + LOG.debug("No disk stats available for detecting outliers."); + continue; + } - detectAndUpdateDiskOutliers(metadataOpStats, readIoStats, - writeIoStats); + detectAndUpdateDiskOutliers(metadataOpStats, readIoStats, + writeIoStats); + } try { Thread.sleep(detectionInterval); @@ -143,9 +149,10 @@ public class DataNodeDiskMetrics { for (Map.Entry entry : writeIoOutliers.entrySet()) { addDiskStat(diskStats, entry.getKey(), DiskOp.WRITE, entry.getValue()); } - - diskOutliersStats = diskStats; - LOG.debug("Updated disk outliers."); + if (overrideStatus) { + diskOutliersStats = diskStats; + LOG.debug("Updated disk outliers."); + } } private void addDiskStat(Map> diskStats, @@ -176,6 +183,7 @@ public class DataNodeDiskMetrics { @VisibleForTesting public void addSlowDiskForTesting(String slowDiskPath, Map latencies) { + overrideStatus = false; if (latencies == null) { diskOutliersStats.put(slowDiskPath, ImmutableMap.of()); } else {