Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: throttled tx metrics #16130

Merged
merged 11 commits into from
Oct 25, 2024
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@
import com.swirlds.common.metrics.IntegerPairAccumulator;
import com.swirlds.common.metrics.RunningAverageMetric;
import com.swirlds.common.metrics.RunningAverageMetric.Config;
import com.swirlds.metrics.api.Counter;
import com.swirlds.metrics.api.IntegerAccumulator;
import com.swirlds.metrics.api.Metrics;
import edu.umd.cs.findbugs.annotations.NonNull;
Expand All @@ -45,7 +46,10 @@
.withDescription("average EVM gas used per second of consensus time")
.withFormat("%,13.6f");

private final Map<HederaFunctionality, TransactionMetric> transactionMetrics =
private final Map<HederaFunctionality, TransactionMetric> transactionDurationMetrics =
new EnumMap<>(HederaFunctionality.class);

private final Map<HederaFunctionality, Counter> transactionThrottleMetrics =
new EnumMap<>(HederaFunctionality.class);

private final RunningAverageMetric gasPerConsSec;
Expand All @@ -68,6 +72,8 @@
}
final var protoName = functionality.protoName();
final var name = protoName.substring(0, 1).toLowerCase() + protoName.substring(1);

// initialize the transaction duration metrics
final var maxConfig = new IntegerAccumulator.Config("app", name + "DurationMax")
.withDescription("The maximum duration of a " + name + " transaction in nanoseconds")
.withUnit("ns");
Expand All @@ -77,15 +83,20 @@
.withDescription("The average duration of a " + name + " transaction in nanoseconds")
.withUnit("ns");
final var avgMetric = metrics.getOrCreate(avgConfig);
transactionMetrics.put(functionality, new TransactionMetric(maxMetric, avgMetric));
transactionDurationMetrics.put(functionality, new TransactionMetric(maxMetric, avgMetric));

// initialize the transaction throttle metrics
final var throttledConfig = new Counter.Config("app", name + "ThrottledTps")
.withDescription("The number of " + name + " transactions that failed due to throttling");
mhess-swl marked this conversation as resolved.
Show resolved Hide resolved
transactionThrottleMetrics.put(functionality, metrics.getOrCreate(throttledConfig));
}

final StatsConfig statsConfig = configProvider.getConfiguration().getConfigData(StatsConfig.class);
gasPerConsSec = metrics.getOrCreate(GAS_PER_CONS_SEC_CONFIG.withHalfLife(statsConfig.runningAvgHalfLifeSecs()));
}

/**
* Update the metrics for the given functionality
* Update the transaction duration metrics for the given functionality
*
* @param functionality the {@link HederaFunctionality} for which the metrics will be updated
* @param duration the duration of the operation in {@code ns}
Expand All @@ -95,7 +106,7 @@
if (functionality == HederaFunctionality.NONE) {
return;
}
final var metric = transactionMetrics.get(functionality);
final var metric = transactionDurationMetrics.get(functionality);
if (metric != null) {
// We do not synchronize the update of the metrics. This may lead to a situation where the max value is
// is stored in one reporting interval and the average in another. This is acceptable as synchronizing
Expand All @@ -105,6 +116,23 @@
}
}

/**
* Increment the throttled metrics for the given functionality, to track the number of transactions per second that
* failed due to throttling
*
* @param functionality the {@link HederaFunctionality} for which the throttled metrics will be updated
*/
public void incrementThrottled(@NonNull final HederaFunctionality functionality) {
requireNonNull(functionality, "functionality must not be null");
if (functionality == HederaFunctionality.NONE) {
return;

Check warning on line 128 in hedera-node/hedera-app/src/main/java/com/hedera/node/app/workflows/OpWorkflowMetrics.java

View check run for this annotation

Codecov / codecov/patch

hedera-node/hedera-app/src/main/java/com/hedera/node/app/workflows/OpWorkflowMetrics.java#L128

Added line #L128 was not covered by tests
}
final var metric = transactionThrottleMetrics.get(functionality);
if (metric != null) {
metric.increment();
}
}

public void switchConsensusSecond() {
gasPerConsSec.update(gasUsedThisConsensusSecond);
gasUsedThisConsensusSecond = 0L;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@
import com.hedera.node.app.spi.authorization.Authorizer;
import com.hedera.node.app.spi.workflows.HandleException;
import com.hedera.node.app.spi.workflows.record.StreamBuilder;
import com.hedera.node.app.workflows.OpWorkflowMetrics;
import com.hedera.node.app.workflows.dispatcher.TransactionDispatcher;
import com.hedera.node.app.workflows.handle.dispatch.DispatchValidator;
import com.hedera.node.app.workflows.handle.dispatch.RecordFinalizer;
Expand Down Expand Up @@ -78,6 +79,7 @@ public class DispatchProcessor {
private final TransactionDispatcher dispatcher;
private final EthereumTransactionHandler ethereumTransactionHandler;
private final NetworkInfo networkInfo;
private final OpWorkflowMetrics workflowMetrics;

@Inject
public DispatchProcessor(
Expand All @@ -90,7 +92,8 @@ public DispatchProcessor(
@NonNull final ExchangeRateManager exchangeRateManager,
@NonNull final TransactionDispatcher dispatcher,
@NonNull final EthereumTransactionHandler ethereumTransactionHandler,
final NetworkInfo networkInfo) {
@NonNull final NetworkInfo networkInfo,
@NonNull final OpWorkflowMetrics workflowMetrics) {
this.authorizer = requireNonNull(authorizer);
this.validator = requireNonNull(validator);
this.recordFinalizer = requireNonNull(recordFinalizer);
Expand All @@ -101,6 +104,7 @@ public DispatchProcessor(
this.dispatcher = requireNonNull(dispatcher);
this.ethereumTransactionHandler = requireNonNull(ethereumTransactionHandler);
this.networkInfo = requireNonNull(networkInfo);
this.workflowMetrics = requireNonNull(workflowMetrics);
}

/**
Expand Down Expand Up @@ -140,7 +144,6 @@ public void processDispatch(@NonNull final Dispatch dispatch) {
*
* @param dispatch the dispatch to be processed
* @param validationResult the due diligence report for the dispatch
* @return the work done by the dispatch
*/
private void tryHandle(@NonNull final Dispatch dispatch, @NonNull final ValidationResult validationResult) {
try {
Expand All @@ -162,8 +165,10 @@ private void tryHandle(@NonNull final Dispatch dispatch, @NonNull final Validati
// Since there is no easy way to say how much work was done in the failed dispatch,
// and current throttling is very rough-grained, we just return USER_TRANSACTION here
} catch (final ThrottleException e) {
var functionality = dispatch.txnInfo().functionality();
workflowMetrics.incrementThrottled(functionality);
rollbackAndRechargeFee(dispatch, validationResult, e.getStatus());
if (dispatch.txnInfo().functionality() == ETHEREUM_TRANSACTION) {
if (functionality == ETHEREUM_TRANSACTION) {
ethereumTransactionHandler.handleThrottled(dispatch.handleContext());
}
} catch (final Exception e) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@
import com.hedera.node.app.state.DeduplicationCache;
import com.hedera.node.app.store.ReadableStoreFactory;
import com.hedera.node.app.throttle.SynchronizedThrottleAccumulator;
import com.hedera.node.app.workflows.OpWorkflowMetrics;
import com.hedera.node.app.workflows.SolvencyPreCheck;
import com.hedera.node.app.workflows.TransactionChecker;
import com.hedera.node.app.workflows.TransactionChecker.RequireMinValidLifetimeBuffer;
Expand Down Expand Up @@ -97,6 +98,7 @@ public final class IngestChecker {
private final Authorizer authorizer;
private final SynchronizedThrottleAccumulator synchronizedThrottleAccumulator;
private final InstantSource instantSource;
private final OpWorkflowMetrics workflowMetrics;

/**
* Constructor of the {@code IngestChecker}
Expand All @@ -111,6 +113,7 @@ public final class IngestChecker {
* @param feeManager the {@link FeeManager} that manages {@link com.hedera.node.app.spi.fees.FeeCalculator}s
* @param synchronizedThrottleAccumulator the {@link SynchronizedThrottleAccumulator} that checks transaction should be throttled
* @param instantSource the {@link InstantSource} that provides the current time
* @param workflowMetrics the {@link OpWorkflowMetrics} that manages the metrics for all operations
* @throws NullPointerException if one of the arguments is {@code null}
*/
@Inject
Expand All @@ -126,7 +129,8 @@ public IngestChecker(
@NonNull final FeeManager feeManager,
@NonNull final Authorizer authorizer,
@NonNull final SynchronizedThrottleAccumulator synchronizedThrottleAccumulator,
@NonNull final InstantSource instantSource) {
@NonNull final InstantSource instantSource,
@NonNull final OpWorkflowMetrics workflowMetrics) {
this.nodeAccount = requireNonNull(nodeAccount, "nodeAccount must not be null");
this.currentPlatformStatus = requireNonNull(currentPlatformStatus, "currentPlatformStatus must not be null");
this.transactionChecker = requireNonNull(transactionChecker, "transactionChecker must not be null");
Expand All @@ -139,6 +143,7 @@ public IngestChecker(
this.authorizer = requireNonNull(authorizer, "authorizer must not be null");
this.synchronizedThrottleAccumulator = requireNonNull(synchronizedThrottleAccumulator);
this.instantSource = requireNonNull(instantSource);
this.workflowMetrics = requireNonNull(workflowMetrics);
}

/**
Expand Down Expand Up @@ -193,10 +198,9 @@ public TransactionInfo runAllChecks(
// 4. Check throttles
assertThrottlingPreconditions(txInfo, configuration);
final var hederaConfig = configuration.getConfigData(HederaConfig.class);
if (hederaConfig.ingestThrottleEnabled()) {
if (synchronizedThrottleAccumulator.shouldThrottle(txInfo, state)) {
throw new PreCheckException(BUSY);
}
if (hederaConfig.ingestThrottleEnabled() && synchronizedThrottleAccumulator.shouldThrottle(txInfo, state)) {
workflowMetrics.incrementThrottled(functionality);
throw new PreCheckException(BUSY);
mhess-swl marked this conversation as resolved.
Show resolved Hide resolved
}

// 4a. Run pure checks
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -124,6 +124,7 @@ public final class QueryWorkflowImpl implements QueryWorkflow {
* @param feeManager the {@link FeeManager} to calculate the fees
* @param synchronizedThrottleAccumulator the {@link SynchronizedThrottleAccumulator} that checks transaction should be throttled
* @param instantSource the {@link InstantSource} to get the current time
* @param workflowMetrics the {@link OpWorkflowMetrics} to update the metrics
* @param shouldCharge If the workflow should charge for handling queries.
* @throws NullPointerException if one of the arguments is {@code null}
*/
Expand Down Expand Up @@ -270,6 +271,7 @@ public void handleQuery(@NonNull final Bytes requestBuffer, @NonNull final Buffe

// 5. Check query throttles
if (shouldCharge && synchronizedThrottleAccumulator.shouldThrottle(function, query, payerID)) {
workflowMetrics.incrementThrottled(function);
throw new PreCheckException(BUSY);
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -69,6 +69,7 @@
import com.hedera.node.app.spi.signatures.SignatureVerification;
import com.hedera.node.app.spi.workflows.HandleContext;
import com.hedera.node.app.spi.workflows.HandleException;
import com.hedera.node.app.workflows.OpWorkflowMetrics;
import com.hedera.node.app.workflows.TransactionInfo;
import com.hedera.node.app.workflows.dispatcher.TransactionDispatcher;
import com.hedera.node.app.workflows.handle.dispatch.DispatchValidator;
Expand Down Expand Up @@ -168,6 +169,9 @@ class DispatchProcessorTest {
@Mock
private NetworkInfo networkInfo;

@Mock
private OpWorkflowMetrics opWorkflowMetrics;

private DispatchProcessor subject;

@BeforeEach
Expand All @@ -182,7 +186,8 @@ void setUp() {
exchangeRateManager,
dispatcher,
ethereumTransactionHandler,
networkInfo);
networkInfo,
opWorkflowMetrics);
given(dispatch.stack()).willReturn(stack);
given(dispatch.recordBuilder()).willReturn(recordBuilder);
}
Expand All @@ -200,6 +205,7 @@ void creatorErrorAsExpected() {
verify(feeAccumulator).chargeNetworkFee(CREATOR_ACCOUNT_ID, FEES.networkFee());
verify(recordBuilder).status(INVALID_PAYER_SIGNATURE);
assertFinished(IsRootStack.NO);
verify(opWorkflowMetrics, never()).incrementThrottled(any());
}

@Test
Expand All @@ -221,6 +227,8 @@ void waivedFeesDoesNotCharge() {
verifyNoInteractions(feeAccumulator);
verify(dispatcher).dispatchHandle(context);
verify(recordBuilder).status(SUCCESS);
verify(opWorkflowMetrics, never()).incrementThrottled(any());

assertFinished();
}

Expand All @@ -240,6 +248,8 @@ void unauthorizedSystemDeleteIsNotSupported() {
verifyTrackedFeePayments();
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES);
verify(recordBuilder).status(NOT_SUPPORTED);
verify(opWorkflowMetrics, never()).incrementThrottled(any());

assertFinished();
}

Expand All @@ -259,6 +269,8 @@ void unauthorizedOtherIsUnauthorized() {
verifyTrackedFeePayments();
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES);
verify(recordBuilder).status(UNAUTHORIZED);
verify(opWorkflowMetrics, never()).incrementThrottled(any());

assertFinished();
}

Expand All @@ -280,6 +292,8 @@ void unprivilegedSystemUndeleteIsAuthorizationFailed() {
verifyTrackedFeePayments();
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES);
verify(recordBuilder).status(AUTHORIZATION_FAILED);
verify(opWorkflowMetrics, never()).incrementThrottled(any());

assertFinished();
}

Expand All @@ -301,6 +315,8 @@ void unprivilegedSystemDeleteIsImpermissible() {
verifyTrackedFeePayments();
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES);
verify(recordBuilder).status(ENTITY_NOT_ALLOWED_TO_DELETE);
verify(opWorkflowMetrics, never()).incrementThrottled(any());

assertFinished();
}

Expand All @@ -322,6 +338,8 @@ void invalidSignatureCryptoTransferFails() {
verifyTrackedFeePayments();
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES);
verify(recordBuilder).status(INVALID_SIGNATURE);
verify(opWorkflowMetrics, never()).incrementThrottled(any());

assertFinished();
}

Expand All @@ -345,6 +363,8 @@ void invalidHollowAccountCryptoTransferFails() {
verifyTrackedFeePayments();
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES);
verify(recordBuilder).status(INVALID_SIGNATURE);
verify(opWorkflowMetrics, never()).incrementThrottled(any());

assertFinished();
}

Expand All @@ -368,6 +388,8 @@ void thrownHandleExceptionRollsBackIfRequested() {
verify(dispatcher).dispatchHandle(context);
verify(recordBuilder).status(TOKEN_NOT_ASSOCIATED_TO_ACCOUNT);
verify(feeAccumulator, times(2)).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES);
verify(opWorkflowMetrics, never()).incrementThrottled(any());

assertFinished();
}

Expand All @@ -391,6 +413,7 @@ void thrownHandleExceptionDoesNotRollBackIfNotRequested() {
verify(dispatcher).dispatchHandle(context);
verify(recordBuilder).status(CONTRACT_REVERT_EXECUTED);
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES);
verify(opWorkflowMetrics, never()).incrementThrottled(any());
assertFinished();
}

Expand All @@ -414,6 +437,7 @@ void consGasExhaustedWaivesServiceFee() throws ThrottleException {
verify(recordBuilder).status(CONSENSUS_GAS_EXHAUSTED);
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES);
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES.withoutServiceComponent());
verify(opWorkflowMetrics).incrementThrottled(CONTRACT_CALL);
mhess-swl marked this conversation as resolved.
Show resolved Hide resolved
assertFinished();
}

Expand All @@ -439,6 +463,7 @@ void consGasExhaustedForEthTxnDoesExtraWork() throws ThrottleException {
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES);
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES.withoutServiceComponent());
verify(ethereumTransactionHandler).handleThrottled(context);
verify(opWorkflowMetrics).incrementThrottled(ETHEREUM_TRANSACTION);
assertFinished();
}

Expand All @@ -460,6 +485,7 @@ void failInvalidWaivesServiceFee() {
verify(recordBuilder).status(FAIL_INVALID);
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES);
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES.withoutServiceComponent());
verify(opWorkflowMetrics, never()).incrementThrottled(any());
assertFinished();
}

Expand All @@ -486,6 +512,7 @@ void happyPathContractCallAsExpected() {
verify(platformStateUpdates).handleTxBody(stack, CONTRACT_TXN_INFO.txBody());
verify(recordBuilder, times(2)).status(SUCCESS);
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES);
verify(opWorkflowMetrics, never()).incrementThrottled(any());
assertFinished();
}

Expand All @@ -505,6 +532,7 @@ void happyPathChildCryptoTransferAsExpected() {
verify(platformStateUpdates, never()).handleTxBody(stack, CRYPTO_TRANSFER_TXN_INFO.txBody());
verify(recordBuilder).status(SUCCESS);
verify(feeAccumulator).chargeNetworkFee(PAYER_ACCOUNT_ID, FEES.totalFee());
verify(opWorkflowMetrics, never()).incrementThrottled(any());
assertFinished(IsRootStack.NO);
}

Expand All @@ -522,6 +550,7 @@ void happyPathFreeChildCryptoTransferAsExpected() {

verify(platformStateUpdates, never()).handleTxBody(stack, CRYPTO_TRANSFER_TXN_INFO.txBody());
verify(recordBuilder).status(SUCCESS);
verify(opWorkflowMetrics, never()).incrementThrottled(any());
assertFinished(IsRootStack.NO);
}

Expand All @@ -545,6 +574,7 @@ void unableToAffordServiceFeesChargesAccordingly() {
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES.withoutServiceComponent());
verify(recordBuilder).status(INSUFFICIENT_ACCOUNT_BALANCE);
verifyNoInteractions(dispatcher);
verify(opWorkflowMetrics, never()).incrementThrottled(any());
assertFinished();
}

Expand All @@ -564,6 +594,7 @@ void duplicateChargesAccordingly() {
verify(feeAccumulator).chargeFees(PAYER_ACCOUNT_ID, CREATOR_ACCOUNT_ID, FEES.withoutServiceComponent());
verify(recordBuilder).status(DUPLICATE_TRANSACTION);
verifyNoInteractions(dispatcher);
verify(opWorkflowMetrics, never()).incrementThrottled(any());
assertFinished();
}

Expand Down
Loading
Loading