[jira] [HBASE-5744] Thrift server metrics should be long instead of int

Summary:
As we measure our Thrift call latencies in nanoseconds, we need to make
latencies long instead of int everywhere. There is a bug where we truncate a
nanosecond latency to int, which is a problem with RPCs that take more than
2.147483647 seconds to process.

This is a trunk diff. The 89-fb version of this diff is at D2679.

Test Plan:
TestThriftServer is updated to test for the failure case (an RPC is artificially
made to take 3 seconds). The new test case fails without the fix.

Re-run all unit tests.

Reviewers: stack, sc, Kannan, JIRA

Reviewed By: sc

Differential Revision: https://reviews.facebook.net/D2685



git-svn-id: https://svn.apache.org/repos/asf/hbase/trunk@1311167 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
mbautin 2012-04-09 09:00:11 +00:00
parent 720a974794
commit 00316a4a51
3 changed files with 55 additions and 19 deletions

View File

@ -65,7 +65,7 @@ public class HbaseHandlerMetricsProxy implements InvocationHandler {
try { try {
long start = now(); long start = now();
result = m.invoke(handler, args); result = m.invoke(handler, args);
int processTime = (int)(now() - start); long processTime = now() - start;
metrics.incMethodTime(m.getName(), processTime); metrics.incMethodTime(m.getName(), processTime);
} catch (InvocationTargetException e) { } catch (InvocationTargetException e) {
throw e.getTargetException(); throw e.getTargetException();

View File

@ -98,16 +98,16 @@ public class ThriftMetrics implements Updater {
numRowKeysInBatchMutate.inc(diff); numRowKeysInBatchMutate.inc(diff);
} }
public void incMethodTime(String name, int time) { public void incMethodTime(String name, long time) {
MetricsTimeVaryingRate methodTimeMetrc = getMethodTimeMetrics(name); MetricsTimeVaryingRate methodTimeMetric = getMethodTimeMetrics(name);
if (methodTimeMetrc == null) { if (methodTimeMetric == null) {
LOG.warn( LOG.warn(
"Got incMethodTime() request for method that doesnt exist: " + name); "Got incMethodTime() request for method that doesnt exist: " + name);
return; // ignore methods that dont exist. return; // ignore methods that dont exist.
} }
// inc method specific processTime // inc method specific processTime
methodTimeMetrc.inc(time); methodTimeMetric.inc(time);
// inc general processTime // inc general processTime
thriftCall.inc(time); thriftCall.inc(time);

View File

@ -40,11 +40,13 @@ import org.apache.hadoop.hbase.filter.ParseFilter;
import org.apache.hadoop.hbase.thrift.generated.BatchMutation; import org.apache.hadoop.hbase.thrift.generated.BatchMutation;
import org.apache.hadoop.hbase.thrift.generated.ColumnDescriptor; import org.apache.hadoop.hbase.thrift.generated.ColumnDescriptor;
import org.apache.hadoop.hbase.thrift.generated.Hbase; import org.apache.hadoop.hbase.thrift.generated.Hbase;
import org.apache.hadoop.hbase.thrift.generated.IOError;
import org.apache.hadoop.hbase.thrift.generated.Mutation; import org.apache.hadoop.hbase.thrift.generated.Mutation;
import org.apache.hadoop.hbase.thrift.generated.TCell; import org.apache.hadoop.hbase.thrift.generated.TCell;
import org.apache.hadoop.hbase.thrift.generated.TRegionInfo; import org.apache.hadoop.hbase.thrift.generated.TRegionInfo;
import org.apache.hadoop.hbase.thrift.generated.TRowResult; import org.apache.hadoop.hbase.thrift.generated.TRowResult;
import org.apache.hadoop.hbase.util.Bytes; import org.apache.hadoop.hbase.util.Bytes;
import org.apache.hadoop.hbase.util.Threads;
import org.apache.hadoop.metrics.ContextFactory; import org.apache.hadoop.metrics.ContextFactory;
import org.apache.hadoop.metrics.MetricsContext; import org.apache.hadoop.metrics.MetricsContext;
import org.apache.hadoop.metrics.MetricsUtil; import org.apache.hadoop.metrics.MetricsUtil;
@ -130,24 +132,43 @@ public class TestThriftServer {
dropTestTables(handler); dropTestTables(handler);
} }
public static final class MySlowHBaseHandler extends ThriftServerRunner.HBaseHandler
implements Hbase.Iface {
protected MySlowHBaseHandler(Configuration c)
throws IOException {
super(c);
}
@Override
public List<ByteBuffer> getTableNames() throws IOError {
Threads.sleepWithoutInterrupt(3000);
return super.getTableNames();
}
}
/** /**
* Tests if the metrics for thrift handler work correctly * Tests if the metrics for thrift handler work correctly
*/ */
public void doTestThriftMetrics() throws Exception { public void doTestThriftMetrics() throws Exception {
Configuration conf = UTIL.getConfiguration(); Configuration conf = UTIL.getConfiguration();
ThriftMetrics metrics = getMetrics(conf); ThriftMetrics metrics = getMetrics(conf);
Hbase.Iface handler = getHandler(metrics, conf); Hbase.Iface handler = getHandlerForMetricsTest(metrics, conf);
createTestTables(handler); createTestTables(handler);
dropTestTables(handler); dropTestTables(handler);
verifyMetrics(metrics, "createTable_num_ops", 2); verifyMetrics(metrics, "createTable_num_ops", 2);
verifyMetrics(metrics, "deleteTable_num_ops", 2); verifyMetrics(metrics, "deleteTable_num_ops", 2);
verifyMetrics(metrics, "disableTable_num_ops", 2); verifyMetrics(metrics, "disableTable_num_ops", 2);
handler.getTableNames(); // This will have an artificial delay.
// 3 to 6 seconds (to account for potential slowness), measured in nanoseconds.
verifyMetricRange(metrics, "getTableNames_avg_time", 3L * 1000 * 1000 * 1000,
6L * 1000 * 1000 * 1000);
} }
private static Hbase.Iface getHandler(ThriftMetrics metrics, Configuration conf) private static Hbase.Iface getHandlerForMetricsTest(ThriftMetrics metrics, Configuration conf)
throws Exception { throws Exception {
Hbase.Iface handler = Hbase.Iface handler = new MySlowHBaseHandler(conf);
new ThriftServerRunner.HBaseHandler(conf);
return HbaseHandlerMetricsProxy.newInstance(handler, metrics, conf); return HbaseHandlerMetricsProxy.newInstance(handler, metrics, conf);
} }
@ -164,16 +185,6 @@ public class TestThriftServer {
.createRecord(ThriftMetrics.CONTEXT_NAME).remove(); .createRecord(ThriftMetrics.CONTEXT_NAME).remove();
} }
private static void verifyMetrics(ThriftMetrics metrics, String name, int expectValue)
throws Exception {
MetricsContext context = MetricsUtil.getContext(
ThriftMetrics.CONTEXT_NAME);
metrics.doUpdates(context);
OutputRecord record = context.getAllRecords().get(
ThriftMetrics.CONTEXT_NAME).iterator().next();
assertEquals(expectValue, record.getMetric(name).intValue());
}
public static void createTestTables(Hbase.Iface handler) throws Exception { public static void createTestTables(Hbase.Iface handler) throws Exception {
// Create/enable/disable/delete tables, ensure methods act correctly // Create/enable/disable/delete tables, ensure methods act correctly
assertEquals(handler.getTableNames().size(), 0); assertEquals(handler.getTableNames().size(), 0);
@ -201,6 +212,31 @@ public class TestThriftServer {
assertEquals(handler.getTableNames().size(), 0); assertEquals(handler.getTableNames().size(), 0);
} }
private static void verifyMetrics(ThriftMetrics metrics, String name, long expectValue)
throws Exception {
long metricVal = getMetricValue(metrics, name);
assertEquals(expectValue, metricVal);
}
private static void verifyMetricRange(ThriftMetrics metrics, String name,
long minValue, long maxValue)
throws Exception {
long metricVal = getMetricValue(metrics, name);
if (metricVal < minValue || metricVal > maxValue) {
throw new AssertionError("Value of metric " + name + " is outside of the expected " +
"range [" + minValue + ", " + maxValue + "]: " + metricVal);
}
}
private static long getMetricValue(ThriftMetrics metrics, String name) {
MetricsContext context = MetricsUtil.getContext(
ThriftMetrics.CONTEXT_NAME);
metrics.doUpdates(context);
OutputRecord record = context.getAllRecords().get(
ThriftMetrics.CONTEXT_NAME).iterator().next();
return record.getMetric(name).longValue();
}
/** /**
* Tests adding a series of Mutations and BatchMutations, including a * Tests adding a series of Mutations and BatchMutations, including a
* delete mutation. Also tests data retrieval, and getting back multiple * delete mutation. Also tests data retrieval, and getting back multiple