diff --git a/hbase-hadoop-compat/src/main/java/org/apache/hadoop/hbase/master/MetricsMasterWrapper.java b/hbase-hadoop-compat/src/main/java/org/apache/hadoop/hbase/master/MetricsMasterWrapper.java index e6b2bc56e53..cce02cf2838 100644 --- a/hbase-hadoop-compat/src/main/java/org/apache/hadoop/hbase/master/MetricsMasterWrapper.java +++ b/hbase-hadoop-compat/src/main/java/org/apache/hadoop/hbase/master/MetricsMasterWrapper.java @@ -29,6 +29,11 @@ import org.apache.yetus.audience.InterfaceAudience; @InterfaceAudience.Private public interface MetricsMasterWrapper { + /** + * Returns if the master is currently running and is not attempting to shutdown. + */ + boolean isRunning(); + /** * Get ServerName */ diff --git a/hbase-hadoop2-compat/src/main/java/org/apache/hadoop/hbase/master/MetricsMasterSourceImpl.java b/hbase-hadoop2-compat/src/main/java/org/apache/hadoop/hbase/master/MetricsMasterSourceImpl.java index 5b78fe8db14..c340f8e8816 100644 --- a/hbase-hadoop2-compat/src/main/java/org/apache/hadoop/hbase/master/MetricsMasterSourceImpl.java +++ b/hbase-hadoop2-compat/src/main/java/org/apache/hadoop/hbase/master/MetricsMasterSourceImpl.java @@ -82,7 +82,9 @@ public class MetricsMasterSourceImpl MetricsRecordBuilder metricsRecordBuilder = metricsCollector.addRecord(metricsName); // masterWrapper can be null because this function is called inside of init. - if (masterWrapper != null) { + // If the master is already stopped or has initiated a shutdown, no point in registering the + // metrics again. + if (masterWrapper != null && masterWrapper.isRunning()) { metricsRecordBuilder .addGauge(Interns.info(MERGE_PLAN_COUNT_NAME, MERGE_PLAN_COUNT_DESC), masterWrapper.getMergePlanCount()) diff --git a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/CachedClusterId.java b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/CachedClusterId.java index 9ca73998761..4bc444b90b3 100644 --- a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/CachedClusterId.java +++ b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/CachedClusterId.java @@ -25,6 +25,7 @@ import org.apache.hadoop.conf.Configuration; import org.apache.hadoop.fs.FileSystem; import org.apache.hadoop.fs.Path; import org.apache.hadoop.hbase.ClusterId; +import org.apache.hadoop.hbase.Server; import org.apache.hadoop.hbase.util.FSUtils; import org.apache.yetus.audience.InterfaceAudience; import org.slf4j.Logger; @@ -46,8 +47,8 @@ public class CachedClusterId { public static final Logger LOG = LoggerFactory.getLogger(CachedClusterId.class); private static final int MAX_FETCH_TIMEOUT_MS = 10000; - private Path rootDir; - private FileSystem fs; + private final Path rootDir; + private final FileSystem fs; // When true, indicates that a FileSystem fetch of ClusterID is in progress. This is used to // avoid multiple fetches from FS and let only one thread fetch the information. @@ -58,12 +59,15 @@ public class CachedClusterId { // Immutable once set and read multiple times. private ClusterId clusterId; + private final Server server; + // cache stats for testing. private AtomicInteger cacheMisses = new AtomicInteger(0); - public CachedClusterId(Configuration conf) throws IOException { - rootDir = FSUtils.getRootDir(conf); - fs = rootDir.getFileSystem(conf); + public CachedClusterId(Server server, Configuration conf) throws IOException { + this.rootDir = FSUtils.getRootDir(conf); + this.fs = rootDir.getFileSystem(conf); + this.server = server; } /** @@ -130,9 +134,12 @@ public class CachedClusterId { * trying get from a clean cache. * * @return ClusterId by reading from FileSystem or null in any error case or cluster ID does - * not exist on the file system. + * not exist on the file system or if the server initiated a tear down. */ public String getFromCacheOrFetch() { + if (server.isStopping() || server.isStopped()) { + return null; + } String id = getClusterId(); if (id != null) { return id; diff --git a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/HMaster.java b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/HMaster.java index 232b3abd00b..00bf0f7b151 100644 --- a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/HMaster.java +++ b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/HMaster.java @@ -574,7 +574,7 @@ public class HMaster extends HRegionServer implements MasterServices { this.metaRegionLocationCache = null; this.activeMasterManager = null; } - cachedClusterId = new CachedClusterId(conf); + cachedClusterId = new CachedClusterId(this, conf); } catch (Throwable t) { // Make sure we log the exception. HMaster is often started via reflection and the // cause of failed startup is lost. diff --git a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/MetricsMasterWrapperImpl.java b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/MetricsMasterWrapperImpl.java index 6a9b572936d..4803a49620e 100644 --- a/hbase-server/src/main/java/org/apache/hadoop/hbase/master/MetricsMasterWrapperImpl.java +++ b/hbase-server/src/main/java/org/apache/hadoop/hbase/master/MetricsMasterWrapperImpl.java @@ -129,6 +129,10 @@ public class MetricsMasterWrapperImpl implements MetricsMasterWrapper { return serverManager.getDeadServers().size(); } + @Override public boolean isRunning() { + return !(master.isStopped() || master.isStopping()); + } + @Override public String getServerName() { ServerName serverName = master.getServerName(); diff --git a/hbase-server/src/test/java/org/apache/hadoop/hbase/TestCachedClusterId.java b/hbase-server/src/test/java/org/apache/hadoop/hbase/TestCachedClusterId.java index b8a4be85ba5..c146685699e 100644 --- a/hbase-server/src/test/java/org/apache/hadoop/hbase/TestCachedClusterId.java +++ b/hbase-server/src/test/java/org/apache/hadoop/hbase/TestCachedClusterId.java @@ -76,7 +76,8 @@ public class TestCachedClusterId { @Test public void testMultiThreadedGetClusterId() throws Exception { Configuration conf = TEST_UTIL.getConfiguration(); - CachedClusterId cachedClusterId = new CachedClusterId(conf); + CachedClusterId cachedClusterId = new CachedClusterId(TEST_UTIL.getHBaseCluster().getMaster(), + conf); TestContext context = new TestContext(conf); int numThreads = 16; for (int i = 0; i < numThreads; i++) {