From ce91d35b299a28f1cdcfef7a0ecbf47aafa430c0 Mon Sep 17 00:00:00 2001 From: Mukul Kumar Singh Date: Mon, 8 Jul 2019 12:19:16 +0530 Subject: [PATCH] HDDS-1728. Add metrics for leader's latency in ContainerStateMachine. Contributed by Mukul Kumar Singh. (#1022) --- .../transport/server/ratis/CSMMetrics.java | 30 +++++++++++++++++++ .../server/ratis/ContainerStateMachine.java | 26 ++++++++++++---- .../server/ratis/TestCSMMetrics.java | 6 ++++ 3 files changed, 57 insertions(+), 5 deletions(-) diff --git a/hadoop-hdds/container-service/src/main/java/org/apache/hadoop/ozone/container/common/transport/server/ratis/CSMMetrics.java b/hadoop-hdds/container-service/src/main/java/org/apache/hadoop/ozone/container/common/transport/server/ratis/CSMMetrics.java index 1ae3c53d90..def0d7ff40 100644 --- a/hadoop-hdds/container-service/src/main/java/org/apache/hadoop/ozone/container/common/transport/server/ratis/CSMMetrics.java +++ b/hadoop-hdds/container-service/src/main/java/org/apache/hadoop/ozone/container/common/transport/server/ratis/CSMMetrics.java @@ -19,11 +19,14 @@ package org.apache.hadoop.ozone.container.common.transport.server.ratis; import com.google.common.annotations.VisibleForTesting; import org.apache.hadoop.classification.InterfaceAudience; +import org.apache.hadoop.hdds.protocol.datanode.proto.ContainerProtos; import org.apache.hadoop.metrics2.MetricsSystem; import org.apache.hadoop.metrics2.annotation.Metric; import org.apache.hadoop.metrics2.annotation.Metrics; import org.apache.hadoop.metrics2.lib.DefaultMetricsSystem; import org.apache.hadoop.metrics2.lib.MutableCounterLong; +import org.apache.hadoop.metrics2.lib.MutableRate; +import org.apache.hadoop.metrics2.lib.MetricsRegistry; import org.apache.ratis.protocol.RaftGroupId; /** @@ -43,14 +46,28 @@ public class CSMMetrics { private @Metric MutableCounterLong numBytesWrittenCount; private @Metric MutableCounterLong numBytesCommittedCount; + private @Metric MutableRate transactionLatency; + private MutableRate[] opsLatency; + private MetricsRegistry registry = null; + // Failure Metrics private @Metric MutableCounterLong numWriteStateMachineFails; private @Metric MutableCounterLong numQueryStateMachineFails; private @Metric MutableCounterLong numApplyTransactionFails; private @Metric MutableCounterLong numReadStateMachineFails; private @Metric MutableCounterLong numReadStateMachineMissCount; + private @Metric MutableCounterLong numStartTransactionVerifyFailures; + private @Metric MutableCounterLong numContainerNotOpenVerifyFailures; public CSMMetrics() { + int numCmdTypes = ContainerProtos.Type.values().length; + this.opsLatency = new MutableRate[numCmdTypes]; + this.registry = new MetricsRegistry(CSMMetrics.class.getName()); + for (int i = 0; i < numCmdTypes; i++) { + opsLatency[i] = registry.newRate( + ContainerProtos.Type.forNumber(i + 1).toString(), + ContainerProtos.Type.forNumber(i + 1) + " op"); + } } public static CSMMetrics create(RaftGroupId gid) { @@ -154,6 +171,19 @@ public class CSMMetrics { return numBytesCommittedCount.value(); } + public void incPipelineLatency(ContainerProtos.Type type, long latencyNanos) { + opsLatency[type.ordinal()].add(latencyNanos); + transactionLatency.add(latencyNanos); + } + + public void incNumStartTransactionVerifyFailures() { + numStartTransactionVerifyFailures.incr(); + } + + public void incNumContainerNotOpenVerifyFailures() { + numContainerNotOpenVerifyFailures.incr(); + } + public void unRegister() { MetricsSystem ms = DefaultMetricsSystem.instance(); diff --git a/hadoop-hdds/container-service/src/main/java/org/apache/hadoop/ozone/container/common/transport/server/ratis/ContainerStateMachine.java b/hadoop-hdds/container-service/src/main/java/org/apache/hadoop/ozone/container/common/transport/server/ratis/ContainerStateMachine.java index f4a8008b74..c780429fdc 100644 --- a/hadoop-hdds/container-service/src/main/java/org/apache/hadoop/ozone/container/common/transport/server/ratis/ContainerStateMachine.java +++ b/hadoop-hdds/container-service/src/main/java/org/apache/hadoop/ozone/container/common/transport/server/ratis/ContainerStateMachine.java @@ -25,8 +25,10 @@ import org.apache.commons.io.IOUtils; import org.apache.hadoop.hdds.HddsUtils; import org.apache.hadoop.hdds.protocol.datanode.proto.ContainerProtos; +import org.apache.hadoop.hdds.scm.container.common.helpers.ContainerNotOpenException; import org.apache.hadoop.hdds.scm.container.common.helpers.StorageContainerException; import org.apache.hadoop.ozone.container.common.helpers.ContainerUtils; +import org.apache.hadoop.util.Time; import org.apache.ratis.proto.RaftProtos.RaftPeerRole; import org.apache.ratis.protocol.RaftGroupId; import org.apache.ratis.server.RaftServer; @@ -261,12 +263,19 @@ public class ContainerStateMachine extends BaseStateMachine { @Override public TransactionContext startTransaction(RaftClientRequest request) throws IOException { + long startTime = Time.monotonicNowNanos(); final ContainerCommandRequestProto proto = getContainerCommandRequestProto(request.getMessage().getContent()); Preconditions.checkArgument(request.getRaftGroupId().equals(gid)); try { dispatcher.validateContainerCommand(proto); } catch (IOException ioe) { + if (ioe instanceof ContainerNotOpenException) { + metrics.incNumContainerNotOpenVerifyFailures(); + } else { + metrics.incNumStartTransactionVerifyFailures(); + LOG.error("startTransaction validation failed on leader", ioe); + } TransactionContext ctxt = TransactionContext.newBuilder() .setClientRequest(request) .setStateMachine(this) @@ -296,6 +305,7 @@ public class ContainerStateMachine extends BaseStateMachine { .setClientRequest(request) .setStateMachine(this) .setServerRole(RaftPeerRole.LEADER) + .setStateMachineContext(startTime) .setStateMachineData(write.getData()) .setLogData(commitContainerCommandProto.toByteString()) .build(); @@ -304,6 +314,7 @@ public class ContainerStateMachine extends BaseStateMachine { .setClientRequest(request) .setStateMachine(this) .setServerRole(RaftPeerRole.LEADER) + .setStateMachineContext(startTime) .setLogData(request.getMessage().getContent()) .build(); } @@ -450,8 +461,10 @@ public class ContainerStateMachine extends BaseStateMachine { } private ByteString readStateMachineData( - ContainerCommandRequestProto requestProto, long term, long index) - throws IOException { + ContainerCommandRequestProto requestProto, long term, long index) { + // the stateMachine data is not present in the stateMachine cache, + // increment the stateMachine cache miss count + metrics.incNumReadStateMachineMissCount(); WriteChunkRequestProto writeChunkRequestProto = requestProto.getWriteChunk(); ContainerProtos.ChunkInfo chunkInfo = writeChunkRequestProto.getChunkData(); @@ -526,9 +539,6 @@ public class ContainerStateMachine extends BaseStateMachine { return CompletableFuture.completedFuture(ByteString.EMPTY); } try { - // the stateMachine data is not present in the stateMachine cache, - // increment the stateMachine cache miss count - metrics.incNumReadStateMachineMissCount(); final ContainerCommandRequestProto requestProto = getContainerCommandRequestProto( entry.getStateMachineLogEntry().getLogData()); @@ -621,6 +631,12 @@ public class ContainerStateMachine extends BaseStateMachine { getCommandExecutor(requestProto)); future.thenAccept(m -> { + if (trx.getServerRole() == RaftPeerRole.LEADER) { + long startTime = (long) trx.getStateMachineContext(); + metrics.incPipelineLatency(cmdType, + Time.monotonicNowNanos() - startTime); + } + final Long previous = applyTransactionCompletionMap .put(index, trx.getLogEntry().getTerm()); diff --git a/hadoop-ozone/integration-test/src/test/java/org/apache/hadoop/ozone/container/common/transport/server/ratis/TestCSMMetrics.java b/hadoop-ozone/integration-test/src/test/java/org/apache/hadoop/ozone/container/common/transport/server/ratis/TestCSMMetrics.java index 5d886c1caa..4853059a52 100644 --- a/hadoop-ozone/integration-test/src/test/java/org/apache/hadoop/ozone/container/common/transport/server/ratis/TestCSMMetrics.java +++ b/hadoop-ozone/integration-test/src/test/java/org/apache/hadoop/ozone/container/common/transport/server/ratis/TestCSMMetrics.java @@ -115,6 +115,9 @@ public class TestCSMMetrics { assertCounter("NumApplyTransactionOps", 0L, metric); assertCounter("NumBytesWrittenCount", 0L, metric); assertCounter("NumBytesCommittedCount", 0L, metric); + assertCounter("NumStartTransactionVerifyFailures", 0L, metric); + assertCounter("NumContainerNotOpenVerifyFailures", 0L, metric); + assertCounter("WriteChunkNumOps", 0L, metric); // Write Chunk BlockID blockID = ContainerTestHelper.getTestBlockID(ContainerTestHelper. @@ -133,6 +136,9 @@ public class TestCSMMetrics { assertCounter("NumBytesWrittenCount", 1024L, metric); assertCounter("NumApplyTransactionOps", 1L, metric); assertCounter("NumBytesCommittedCount", 1024L, metric); + assertCounter("NumStartTransactionVerifyFailures", 0L, metric); + assertCounter("NumContainerNotOpenVerifyFailures", 0L, metric); + assertCounter("WriteChunkNumOps", 1L, metric); //Read Chunk ContainerProtos.ContainerCommandRequestProto readChunkRequest =