HBASE-12911 Client-side metrics

First pass. Plumbs up metrics for each connection instance. Exposes
static information about those connections (zk quorum and root node,
cluster id, user). Exposes basic thread pool utilization gauges for
"meta lookup" and "batch" pools.
This commit is contained in:
Nick Dimiduk 2015-08-10 16:48:48 -07:00
parent ae35f65e9a
commit 06989fd1f9
13 changed files with 484 additions and 3 deletions

View File

@ -126,6 +126,10 @@
<groupId>org.apache.hbase</groupId> <groupId>org.apache.hbase</groupId>
<artifactId>hbase-common</artifactId> <artifactId>hbase-common</artifactId>
</dependency> </dependency>
<dependency>
<groupId>org.apache.hbase</groupId>
<artifactId>hbase-hadoop-compat</artifactId>
</dependency>
<dependency> <dependency>
<groupId>org.apache.hbase</groupId> <groupId>org.apache.hbase</groupId>
<artifactId>hbase-common</artifactId> <artifactId>hbase-common</artifactId>

View File

@ -175,4 +175,11 @@ public interface Connection extends Abortable, Closeable {
*/ */
boolean isClosed(); boolean isClosed();
/**
* Retrieve the metrics instance for this connection.
*
* @return a MetricsConnection instance for examining connection metrics.
*/
public MetricsConnection getConnectionMetrics();
} }

View File

@ -130,6 +130,11 @@ abstract class ConnectionAdapter implements ClusterConnection {
return wrappedConnection.getAdmin(); return wrappedConnection.getAdmin();
} }
@Override
public MetricsConnection getConnectionMetrics() {
return wrappedConnection.getConnectionMetrics();
}
@Override @Override
public boolean isMasterRunning() throws MasterNotRunningException, public boolean isMasterRunning() throws MasterNotRunningException,
ZooKeeperConnectionException { ZooKeeperConnectionException {

View File

@ -118,6 +118,7 @@ class ConnectionImplementation implements ClusterConnection, Closeable {
private final AsyncProcess asyncProcess; private final AsyncProcess asyncProcess;
// single tracker per connection // single tracker per connection
private final ServerStatisticTracker stats; private final ServerStatisticTracker stats;
private final MetricsConnection metrics;
private volatile boolean closed; private volatile boolean closed;
private volatile boolean aborted; private volatile boolean aborted;
@ -154,11 +155,11 @@ class ConnectionImplementation implements ClusterConnection, Closeable {
// Client rpc instance. // Client rpc instance.
private RpcClient rpcClient; private RpcClient rpcClient;
private MetaCache metaCache = new MetaCache(); private final MetaCache metaCache;
private int refCount; private int refCount;
private User user; protected User user;
private RpcRetryingCallerFactory rpcCallerFactory; private RpcRetryingCallerFactory rpcCallerFactory;
@ -236,11 +237,13 @@ class ConnectionImplementation implements ClusterConnection, Closeable {
} else { } else {
nonceGenerator = new NoNonceGenerator(); nonceGenerator = new NoNonceGenerator();
} }
stats = ServerStatisticTracker.create(conf); this.stats = ServerStatisticTracker.create(conf);
this.asyncProcess = createAsyncProcess(this.conf); this.asyncProcess = createAsyncProcess(this.conf);
this.interceptor = (new RetryingCallerInterceptorFactory(conf)).build(); this.interceptor = (new RetryingCallerInterceptorFactory(conf)).build();
this.rpcCallerFactory = RpcRetryingCallerFactory.instantiate(conf, interceptor, this.stats); this.rpcCallerFactory = RpcRetryingCallerFactory.instantiate(conf, interceptor, this.stats);
this.backoffPolicy = ClientBackoffPolicyFactory.create(conf); this.backoffPolicy = ClientBackoffPolicyFactory.create(conf);
this.metrics = new MetricsConnection(new MetricsConnectionWrapperImpl(this));
this.metaCache = new MetaCache(this.metrics);
} }
/** /**
@ -362,6 +365,11 @@ class ConnectionImplementation implements ClusterConnection, Closeable {
return new HBaseAdmin(this); return new HBaseAdmin(this);
} }
@Override
public MetricsConnection getConnectionMetrics() {
return this.metrics;
}
private ExecutorService getBatchPool() { private ExecutorService getBatchPool() {
if (batchPool == null) { if (batchPool == null) {
synchronized (this) { synchronized (this) {

View File

@ -59,6 +59,12 @@ public class MetaCache {
// The access to this attribute must be protected by a lock on cachedRegionLocations // The access to this attribute must be protected by a lock on cachedRegionLocations
private final Set<ServerName> cachedServers = new ConcurrentSkipListSet<ServerName>(); private final Set<ServerName> cachedServers = new ConcurrentSkipListSet<ServerName>();
private final MetricsConnection metrics;
public MetaCache(MetricsConnection metrics) {
this.metrics = metrics;
}
/** /**
* Search the cache for a location that fits our table and row key. * Search the cache for a location that fits our table and row key.
* Return null if no suitable region is located. * Return null if no suitable region is located.
@ -74,6 +80,7 @@ public class MetaCache {
Entry<byte[], RegionLocations> e = tableLocations.floorEntry(row); Entry<byte[], RegionLocations> e = tableLocations.floorEntry(row);
if (e == null) { if (e == null) {
metrics.incrMetaCacheMiss();
return null; return null;
} }
RegionLocations possibleRegion = e.getValue(); RegionLocations possibleRegion = e.getValue();
@ -94,10 +101,12 @@ public class MetaCache {
// HConstants.EMPTY_END_ROW) check itself will pass. // HConstants.EMPTY_END_ROW) check itself will pass.
if (Bytes.equals(endKey, HConstants.EMPTY_END_ROW) || if (Bytes.equals(endKey, HConstants.EMPTY_END_ROW) ||
Bytes.compareTo(endKey, 0, endKey.length, row, 0, row.length) > 0) { Bytes.compareTo(endKey, 0, endKey.length, row, 0, row.length) > 0) {
metrics.incrMetaCacheHit();
return possibleRegion; return possibleRegion;
} }
// Passed all the way through, so we got nothing - complete cache miss // Passed all the way through, so we got nothing - complete cache miss
metrics.incrMetaCacheMiss();
return null; return null;
} }

View File

@ -0,0 +1,48 @@
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.hadoop.hbase.client;
import org.apache.hadoop.hbase.CompatibilitySingletonFactory;
import org.apache.hadoop.hbase.classification.InterfaceAudience;
import org.apache.hadoop.hbase.classification.InterfaceStability;
/**
* This class is for maintaining the various connection statistics and publishing them through
* the metrics interfaces.
*/
@InterfaceStability.Evolving
@InterfaceAudience.Private
public class MetricsConnection {
private final MetricsConnectionWrapper wrapper;
private final MetricsConnectionSource source;
public MetricsConnection(MetricsConnectionWrapper wrapper) {
this.wrapper = wrapper;
this.source = CompatibilitySingletonFactory.getInstance(MetricsConnectionSourceFactory.class)
.createConnection(this.wrapper);
}
public void incrMetaCacheHit() {
source.incrMetaCacheHit();
}
public void incrMetaCacheMiss() {
source.incrMetaCacheMiss();
}
}

View File

@ -0,0 +1,99 @@
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.hadoop.hbase.client;
import com.google.common.base.Preconditions;
import java.util.concurrent.ThreadPoolExecutor;
public class MetricsConnectionWrapperImpl implements MetricsConnectionWrapper {
private final ConnectionImplementation conn;
public MetricsConnectionWrapperImpl(ConnectionImplementation connection) {
Preconditions.checkNotNull(connection);
this.conn = connection;
}
@Override public String getId() {
return conn.toString();
}
@Override public String getUserName() {
return conn.user == null ? "" : conn.user.toString();
}
@Override public String getClusterId() {
return conn.clusterId;
}
@Override public String getZookeeperQuorum() {
try {
return conn.getKeepAliveZooKeeperWatcher().getQuorum();
} catch (Exception e) {
return "";
}
}
@Override public String getZookeeperBaseNode() {
try {
return conn.getKeepAliveZooKeeperWatcher().getBaseZNode();
} catch (Exception e) {
return "";
}
}
@Override public int getMetaLookupPoolActiveCount() {
if (conn.getCurrentMetaLookupPool() == null) {
return 0;
}
ThreadPoolExecutor tpe = (ThreadPoolExecutor) conn.getCurrentMetaLookupPool();
return tpe.getActiveCount();
}
@Override public int getMetaLookupPoolLargestPoolSize() {
if (conn.getCurrentMetaLookupPool() == null) {
return 0;
}
ThreadPoolExecutor tpe = (ThreadPoolExecutor) conn.getCurrentMetaLookupPool();
return tpe.getLargestPoolSize();
}
@Override public String getBatchPoolId() {
if (conn.getCurrentBatchPool() == null) {
return "";
}
return Integer.toHexString(conn.getCurrentBatchPool().hashCode());
}
@Override public int getBatchPoolActiveCount() {
if (conn.getCurrentBatchPool() == null) {
return 0;
}
ThreadPoolExecutor tpe = (ThreadPoolExecutor) conn.getCurrentBatchPool();
return tpe.getActiveCount();
}
@Override public int getBatchPoolLargestPoolSize() {
if (conn.getCurrentBatchPool() == null) {
return 0;
}
ThreadPoolExecutor tpe = (ThreadPoolExecutor) conn.getCurrentBatchPool();
return tpe.getLargestPoolSize();
}
}

View File

@ -0,0 +1,87 @@
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.hadoop.hbase.client;
import org.apache.hadoop.hbase.metrics.BaseSource;
public interface MetricsConnectionSource extends BaseSource {
/**
* The name of the metrics
*/
String METRICS_NAME = "Connection";
/**
* The name of the metrics context that metrics will be under.
*/
String METRICS_CONTEXT = "connection";
/**
* Description
*/
String METRICS_DESCRIPTION = "Metrics about HBase Connection";
/**
* The name of the metrics context that metrics will be under in jmx
*/
String METRICS_JMX_CONTEXT = "Client,sub=";
/**
* Increment number of meta cache hits
*/
void incrMetaCacheHit();
/**
* Increment number of meta cache misses
*/
void incrMetaCacheMiss();
// Strings used for exporting to metrics system.
String CONNECTION_ID_NAME = "connectionId";
String CONNECTION_ID_DESC = "The connection's process-unique identifier.";
String USER_NAME_NAME = "userName";
String USER_NAME_DESC = "The user on behalf of whom the Connection is acting.";
String CLUSTER_ID_NAME = "clusterId";
String CLUSTER_ID_DESC = "Cluster Id";
String ZOOKEEPER_QUORUM_NAME = "zookeeperQuorum";
String ZOOKEEPER_QUORUM_DESC = "Zookeeper Quorum";
String ZOOKEEPER_ZNODE_NAME = "zookeeperBaseZNode";
String ZOOKEEPER_ZNODE_DESC = "Base ZNode for this cluster.";
String META_CACHE_HIT_NAME = "metaCacheHit";
String META_CACHE_HIT_DESC =
"A counter on the number of times this connection's meta cache has a valid region location.";
String META_CACHE_MISS_NAME = "metaCacheMiss";
String META_CACHE_MISS_DESC =
"A counter on the number of times this connection does not know where to find a region.";
String META_LOOKUP_POOL_ACTIVE_THREAD_NAME = "metaLookupPoolActiveThreads";
String META_LOOKUP_POOL_ACTIVE_THREAD_DESC =
"The approximate number of threads actively resolving region locations from META.";
String META_LOOKUP_POOL_LARGEST_SIZE_NAME = "metaLookupPoolLargestSize";
String META_LOOKUP_POOL_LARGEST_SIZE_DESC =
"The largest number of threads that have ever simultaneously been in the pool.";
String BATCH_POOL_ID_NAME = "batchPoolId";
String BATCH_POOL_ID_DESC = "The connection's batch pool's unique identifier.";
String BATCH_POOL_ACTIVE_THREAD_NAME = "batchPoolActiveThreads";
String BATCH_POOL_ACTIVE_THREAD_DESC =
"The approximate number of threads executing table operations.";
String BATCH_POOL_LARGEST_SIZE_NAME = "batchPoolLargestSize";
String BATCH_POOL_LARGEST_SIZE_DESC =
"The largest number of threads that have ever simultaneously been in the pool.";
}

View File

@ -0,0 +1,32 @@
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.hadoop.hbase.client;
/**
* Interface of a factory to create Metrics Sources used inside of Connections.
*/
public interface MetricsConnectionSourceFactory {
/**
* Given a wrapper create a {@link MetricsConnectionSource}.
*
* @param wrapper The wrapped Connection
* @return a Metrics Source.
*/
public MetricsConnectionSource createConnection(MetricsConnectionWrapper wrapper);
}

View File

@ -0,0 +1,50 @@
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.hadoop.hbase.client;
/**
* This is the interface that will expose Connection information to hadoop1/hadoop2
* implementations of the {@link MetricsConnectionSource}.
*/
public interface MetricsConnectionWrapper {
/** Get the connection's unique identifier */
String getId();
/** Get the User's name. */
String getUserName();
/** Get the Cluster ID */
String getClusterId();
/** Get the Zookeeper Quorum Info */
String getZookeeperQuorum();
/** Get the base ZNode for this cluster. */
String getZookeeperBaseNode();
int getMetaLookupPoolActiveCount();
int getMetaLookupPoolLargestPoolSize();
String getBatchPoolId();
int getBatchPoolActiveCount();
int getBatchPoolLargestPoolSize();
}

View File

@ -0,0 +1,28 @@
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.hadoop.hbase.client;
import org.apache.hadoop.hbase.classification.InterfaceAudience;
@InterfaceAudience.Private
public class MetricsConnectionSourceFactoryImpl implements MetricsConnectionSourceFactory {
@Override public MetricsConnectionSource createConnection(MetricsConnectionWrapper wrapper) {
return new MetricsConnectionSourceImpl(wrapper);
}
}

View File

@ -0,0 +1,86 @@
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.hadoop.hbase.client;
import org.apache.hadoop.hbase.metrics.BaseSourceImpl;
import org.apache.hadoop.metrics2.MetricsCollector;
import org.apache.hadoop.metrics2.MetricsRecordBuilder;
import org.apache.hadoop.metrics2.lib.Interns;
import org.apache.hadoop.metrics2.lib.MutableCounterLong;
public class MetricsConnectionSourceImpl
extends BaseSourceImpl implements MetricsConnectionSource {
// wrapper for access statistics collected in Connection instance
private final MetricsConnectionWrapper wrapper;
// Hadoop Metric2 objects for additional monitoring
private final MutableCounterLong metaCacheHit;
private final MutableCounterLong metaCacheMiss;
public MetricsConnectionSourceImpl(MetricsConnectionWrapper wrapper) {
this(METRICS_NAME, METRICS_DESCRIPTION, METRICS_CONTEXT,
METRICS_JMX_CONTEXT + wrapper.getId(), wrapper);
}
public MetricsConnectionSourceImpl(String metricsName, String metricsDescription,
String metricsContext, String metricsJmxContext, MetricsConnectionWrapper wrapper) {
super(metricsName, metricsDescription, metricsContext, metricsJmxContext);
this.wrapper = wrapper;
metaCacheHit = getMetricsRegistry().newCounter(META_CACHE_HIT_NAME, META_CACHE_HIT_DESC, 0l);
metaCacheMiss =
getMetricsRegistry().newCounter(META_CACHE_MISS_NAME, META_CACHE_MISS_DESC, 0l);
}
@Override
public void getMetrics(MetricsCollector metricsCollector, boolean all) {
MetricsRecordBuilder mrb = metricsCollector.addRecord(metricsName);
if (wrapper != null) {
mrb.addGauge(Interns.info(META_LOOKUP_POOL_LARGEST_SIZE_NAME,
META_LOOKUP_POOL_LARGEST_SIZE_DESC), wrapper.getMetaLookupPoolLargestPoolSize())
.addGauge(Interns.info(META_LOOKUP_POOL_ACTIVE_THREAD_NAME,
META_LOOKUP_POOL_ACTIVE_THREAD_DESC), wrapper.getMetaLookupPoolActiveCount())
.tag(Interns.info(BATCH_POOL_ID_NAME, BATCH_POOL_ID_DESC), wrapper.getBatchPoolId())
.addGauge(Interns.info(BATCH_POOL_ACTIVE_THREAD_NAME, BATCH_POOL_ACTIVE_THREAD_DESC),
wrapper.getBatchPoolActiveCount())
.addGauge(Interns.info(BATCH_POOL_LARGEST_SIZE_NAME, BATCH_POOL_LARGEST_SIZE_DESC),
wrapper.getBatchPoolLargestPoolSize())
.tag(Interns.info(CONNECTION_ID_NAME, CONNECTION_ID_DESC), wrapper.getId())
.tag(Interns.info(USER_NAME_NAME, USER_NAME_DESC), wrapper.getUserName())
.tag(Interns.info(CLUSTER_ID_NAME, CLUSTER_ID_DESC), wrapper.getClusterId())
.tag(Interns.info(ZOOKEEPER_QUORUM_NAME, ZOOKEEPER_QUORUM_DESC),
wrapper.getZookeeperQuorum())
.tag(Interns.info(ZOOKEEPER_ZNODE_NAME, ZOOKEEPER_ZNODE_DESC),
wrapper.getZookeeperBaseNode());
}
metricsRegistry.snapshot(mrb, all);
}
@Override public void incrMetaCacheHit() {
metaCacheHit.incr();
}
@Override public void incrMetaCacheMiss() {
metaCacheMiss.incr();
}
}

View File

@ -0,0 +1,18 @@
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements. See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership. The ASF licenses this file
# to you under the Apache License, Version 2.0 (the
# "License"); you may not use this file except in compliance
# with the License. You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing,
# software distributed under the License is distributed on an
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
# KIND, either express or implied. See the License for the
# specific language governing permissions and limitations
# under the License.
#
org.apache.hadoop.hbase.client.MetricsConnectionSourceFactoryImpl