Skip to content
This repository has been archived by the owner on Sep 26, 2019. It is now read-only.

PAN-2057: Accept transactions in the pool with nonce above account sender nonce #1065

Merged
merged 2 commits into from
Mar 7, 2019
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -156,11 +156,7 @@ private ValidationResult<TransactionInvalidReason> validateTransaction(
.map(
worldState -> {
final Account senderAccount = worldState.get(transaction.getSender());
return getTransactionValidator()
.validateForSender(
transaction,
senderAccount,
pendingTransactions.getNextNonceForSender(transaction.getSender()));
return getTransactionValidator().validateForSender(transaction, senderAccount, true);
})
.orElseGet(() -> ValidationResult.invalid(CHAIN_HEAD_WORLD_STATE_NOT_AVAILABLE));
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,6 @@

import java.util.ArrayDeque;
import java.util.Deque;
import java.util.OptionalLong;

import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
Expand Down Expand Up @@ -158,8 +157,7 @@ public Result processTransaction(

final Address senderAddress = transaction.getSender();
final MutableAccount sender = worldState.getOrCreate(senderAddress);
validationResult =
transactionValidator.validateForSender(transaction, sender, OptionalLong.empty());
validationResult = transactionValidator.validateForSender(transaction, sender, false);
if (!validationResult.isValid()) {
LOG.warn("Invalid transaction: {}", validationResult.getErrorMessage());
return Result.invalid(validationResult);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,6 @@
import tech.pegasys.pantheon.ethereum.vm.GasCalculator;

import java.util.OptionalInt;
import java.util.OptionalLong;

/**
* Validates a transaction based on Frontier protocol runtime requirements.
Expand Down Expand Up @@ -86,45 +85,52 @@ public ValidationResult<TransactionInvalidReason> validate(final Transaction tra

@Override
public ValidationResult<TransactionInvalidReason> validateForSender(
final Transaction transaction, final Account sender, final OptionalLong maximumNonce) {
final Transaction transaction, final Account sender, final boolean allowFutureNonce) {

Wei balance = Account.DEFAULT_BALANCE;
long nonce = Account.DEFAULT_NONCE;
Wei senderBalance = Account.DEFAULT_BALANCE;
long senderNonce = Account.DEFAULT_NONCE;

if (sender != null) {
balance = sender.getBalance();
nonce = sender.getNonce();
senderBalance = sender.getBalance();
senderNonce = sender.getNonce();
}

if (transaction.getUpfrontCost().compareTo(balance) > 0) {
if (transaction.getUpfrontCost().compareTo(senderBalance) > 0) {
return ValidationResult.invalid(
UPFRONT_COST_EXCEEDS_BALANCE,
String.format(
"transaction up-front cost %s exceeds transaction sender account balance %s",
transaction.getUpfrontCost(), balance));
transaction.getUpfrontCost(), senderBalance));
}

if (transaction.getNonce() < nonce) {
if (transaction.getNonce() < senderNonce) {
return ValidationResult.invalid(
NONCE_TOO_LOW,
String.format(
"transaction nonce %s below sender account nonce %s", transaction.getNonce(), nonce));
"transaction nonce %s below sender account nonce %s",
transaction.getNonce(), senderNonce));
}

if (violatesMaximumNonce(transaction, maximumNonce) && nonce != transaction.getNonce()) {
return ValidationResult.invalid(
INCORRECT_NONCE,
String.format(
"transaction nonce %s does not match sender account nonce %s.",
transaction.getNonce(), nonce));
if (allowFutureNonce) {
if (senderNonce > transaction.getNonce()) {
return invalidNonceResult(transaction, senderNonce);
}
} else {
if (senderNonce != transaction.getNonce()) {
return invalidNonceResult(transaction, senderNonce);
}
}

return ValidationResult.valid();
}

private boolean violatesMaximumNonce(
final Transaction transaction, final OptionalLong maximumNonce) {
return !maximumNonce.isPresent() || transaction.getNonce() > maximumNonce.getAsLong();
private ValidationResult<TransactionInvalidReason> invalidNonceResult(
final Transaction transaction, final long senderNonce) {
return ValidationResult.invalid(
INCORRECT_NONCE,
String.format(
"transaction nonce %s does not match sender account nonce %s.",
transaction.getNonce(), senderNonce));
}

public ValidationResult<TransactionInvalidReason> validateTransactionSignature(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -15,8 +15,6 @@
import tech.pegasys.pantheon.ethereum.core.Account;
import tech.pegasys.pantheon.ethereum.core.Transaction;

import java.util.OptionalLong;

/** Validates transaction based on some criteria. */
public interface TransactionValidator {

Expand All @@ -38,14 +36,16 @@ public interface TransactionValidator {
*
* @param transaction the transaction to validateMessageFrame.State.COMPLETED_FAILED
* @param sender the sender account state to validate against
* @param maximumNonce the maximum transaction nonce. If not provided the transaction nonce must
* equal the sender's current account nonce
* @param allowFutureNonce if true, transactions with nonce equal or higher than the account nonce
* will be considered valid (used when received transactions in the transaction pool). If
* false, only a transaction with the nonce equals the account nonce will be considered valid
* (used when processing transactions).
* @return An empty @{link Optional} if the transaction is considered valid; otherwise an @{code
* Optional} containing a {@link TransactionInvalidReason} that identifies why the transaction
* is invalid.
*/
ValidationResult<TransactionInvalidReason> validateForSender(
Transaction transaction, Account sender, OptionalLong maximumNonce);
Transaction transaction, Account sender, boolean allowFutureNonce);

enum TransactionInvalidReason {
WRONG_CHAIN_ID,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
import static java.util.stream.Collectors.toList;
import static org.assertj.core.api.Assertions.assertThat;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.ArgumentMatchers.anyBoolean;
import static org.mockito.ArgumentMatchers.anyLong;
import static org.mockito.ArgumentMatchers.eq;
import static org.mockito.ArgumentMatchers.nullable;
Expand All @@ -44,7 +45,6 @@
import tech.pegasys.pantheon.util.uint.UInt256;

import java.util.List;
import java.util.OptionalLong;

import org.junit.Before;
import org.junit.Test;
Expand Down Expand Up @@ -211,7 +211,7 @@ public void shouldNotAddRemoteTransactionsThatAreInvalidAccordingToInvariantChec
public void shouldNotAddRemoteTransactionsThatAreInvalidAccordingToStateDependentChecks() {
givenTransactionIsValid(transaction2);
when(transactionValidator.validate(transaction1)).thenReturn(valid());
when(transactionValidator.validateForSender(transaction1, null, OptionalLong.empty()))
when(transactionValidator.validateForSender(transaction1, null, true))
.thenReturn(ValidationResult.invalid(NONCE_TOO_LOW));

transactionPool.addRemoteTransactions(asList(transaction1, transaction2));
Expand All @@ -230,13 +230,13 @@ public void shouldAllowSequenceOfTransactionsWithIncreasingNonceFromSameSender()

when(transactionValidator.validate(any(Transaction.class))).thenReturn(valid());
when(transactionValidator.validateForSender(
eq(transaction1), nullable(Account.class), eq(OptionalLong.empty())))
eq(transaction1), nullable(Account.class), eq(true)))
.thenReturn(valid());
when(transactionValidator.validateForSender(
eq(transaction2), nullable(Account.class), eq(OptionalLong.of(2))))
eq(transaction2), nullable(Account.class), eq(true)))
.thenReturn(valid());
when(transactionValidator.validateForSender(
eq(transaction3), nullable(Account.class), eq(OptionalLong.of(3))))
eq(transaction3), nullable(Account.class), eq(true)))
.thenReturn(valid());

assertThat(transactionPool.addLocalTransaction(transaction1)).isEqualTo(valid());
Expand All @@ -258,13 +258,13 @@ public void shouldAllowSequenceOfTransactionsWithIncreasingNonceFromSameSender()

when(transactionValidator.validate(any(Transaction.class))).thenReturn(valid());
when(transactionValidator.validateForSender(
eq(transaction1), nullable(Account.class), eq(OptionalLong.empty())))
eq(transaction1), nullable(Account.class), eq(true)))
.thenReturn(valid());
when(transactionValidator.validateForSender(
eq(transaction2), nullable(Account.class), eq(OptionalLong.of(2))))
eq(transaction2), nullable(Account.class), eq(true)))
.thenReturn(valid());
when(transactionValidator.validateForSender(
eq(transaction3), nullable(Account.class), eq(OptionalLong.of(3))))
eq(transaction3), nullable(Account.class), eq(true)))
.thenReturn(valid());

transactionPool.addRemoteTransactions(asList(transaction3, transaction1, transaction2));
Expand All @@ -284,10 +284,10 @@ public void shouldNotNotifyBatchListenerWhenRemoteTransactionDoesNotReplaceExist

when(transactionValidator.validate(any(Transaction.class))).thenReturn(valid());
when(transactionValidator.validateForSender(
eq(transaction1), nullable(Account.class), eq(OptionalLong.empty())))
eq(transaction1), nullable(Account.class), eq(true)))
.thenReturn(valid());
when(transactionValidator.validateForSender(
eq(transaction2), nullable(Account.class), eq(OptionalLong.of(2))))
eq(transaction2), nullable(Account.class), eq(true)))
.thenReturn(valid());

transactionPool.addRemoteTransactions(singletonList(transaction1));
Expand All @@ -308,10 +308,10 @@ public void shouldNotNotifyBatchListenerWhenLocalTransactionDoesNotReplaceExisti

when(transactionValidator.validate(any(Transaction.class))).thenReturn(valid());
when(transactionValidator.validateForSender(
eq(transaction1), nullable(Account.class), eq(OptionalLong.empty())))
eq(transaction1), nullable(Account.class), eq(true)))
.thenReturn(valid());
when(transactionValidator.validateForSender(
eq(transaction2), nullable(Account.class), eq(OptionalLong.of(2))))
eq(transaction2), nullable(Account.class), eq(true)))
.thenReturn(valid());

transactionPool.addLocalTransaction(transaction1);
Expand Down Expand Up @@ -443,7 +443,7 @@ private Transaction createTransaction(final int transactionNumber) {
private void givenTransactionIsValid(final Transaction transaction) {
when(transactionValidator.validate(transaction)).thenReturn(valid());
when(transactionValidator.validateForSender(
eq(transaction), nullable(Account.class), any(OptionalLong.class)))
eq(transaction), nullable(Account.class), anyBoolean()))
.thenReturn(valid());
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -31,8 +31,6 @@
import tech.pegasys.pantheon.ethereum.core.Wei;
import tech.pegasys.pantheon.ethereum.vm.GasCalculator;

import java.util.OptionalLong;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.Mock;
Expand Down Expand Up @@ -80,7 +78,7 @@ public void shouldRejectTransactionWhenTransactionHasIncorrectChainId() {
public void shouldRejectTransactionWhenSenderAccountDoesNotExist() {
final MainnetTransactionValidator validator =
new MainnetTransactionValidator(gasCalculator, false, 1);
assertThat(validator.validateForSender(basicTransaction, null, OptionalLong.empty()))
assertThat(validator.validateForSender(basicTransaction, null, false))
.isEqualTo(ValidationResult.invalid(UPFRONT_COST_EXCEEDS_BALANCE));
}

Expand All @@ -90,43 +88,29 @@ public void shouldRejectTransactionWhenTransactionNonceBelowAccountNonce() {
new MainnetTransactionValidator(gasCalculator, false, 1);

final Account account = accountWithNonce(basicTransaction.getNonce() + 1);
assertThat(validator.validateForSender(basicTransaction, account, OptionalLong.empty()))
assertThat(validator.validateForSender(basicTransaction, account, false))
.isEqualTo(ValidationResult.invalid(NONCE_TOO_LOW));
}

@Test
public void shouldRejectTransactionWhenTransactionNonceAboveAccountNonce() {
public void
shouldRejectTransactionWhenTransactionNonceAboveAccountNonceAndFutureNonceIsNotAllowed() {
final MainnetTransactionValidator validator =
new MainnetTransactionValidator(gasCalculator, false, 1);

final Account account = accountWithNonce(basicTransaction.getNonce() - 1);
assertThat(validator.validateForSender(basicTransaction, account, OptionalLong.empty()))
assertThat(validator.validateForSender(basicTransaction, account, false))
.isEqualTo(ValidationResult.invalid(INCORRECT_NONCE));
}

@Test
public void shouldAcceptTransactionWhenNonceBetweenAccountNonceAndMaximumAllowedNonce() {
public void
shouldAcceptTransactionWhenTransactionNonceAboveAccountNonceAndFutureNonceIsAllowed() {
final MainnetTransactionValidator validator =
new MainnetTransactionValidator(gasCalculator, false, 1);

final Transaction transaction =
new TransactionTestFixture().nonce(10).createTransaction(senderKeys);
final Account account = accountWithNonce(5);

assertThat(validator.validateForSender(transaction, account, OptionalLong.of(15)))
.isEqualTo(ValidationResult.valid());
}

@Test
public void shouldAcceptTransactionWhenNonceEqualsMaximumAllowedNonce() {
final MainnetTransactionValidator validator =
new MainnetTransactionValidator(gasCalculator, false, 1);

final Transaction transaction =
new TransactionTestFixture().nonce(10).createTransaction(senderKeys);
final Account account = accountWithNonce(5);

assertThat(validator.validateForSender(transaction, account, OptionalLong.of(10)))
final Account account = accountWithNonce(basicTransaction.getNonce() - 1);
assertThat(validator.validateForSender(basicTransaction, account, true))
.isEqualTo(ValidationResult.valid());
}

Expand All @@ -139,7 +123,7 @@ public void shouldRejectTransactionWhenNonceExceedsMaximumAllowedNonce() {
new TransactionTestFixture().nonce(11).createTransaction(senderKeys);
final Account account = accountWithNonce(5);

assertThat(validator.validateForSender(transaction, account, OptionalLong.of(10)))
assertThat(validator.validateForSender(transaction, account, false))
.isEqualTo(ValidationResult.invalid(INCORRECT_NONCE));
}

Expand All @@ -153,9 +137,7 @@ public void transactionWithNullSenderCanBeValidIfGasPriceAndValueIsZero() {
final Address arbitrarySender = Address.fromHexString("1");
builder.gasPrice(Wei.ZERO).nonce(0).sender(arbitrarySender).value(Wei.ZERO);

assertThat(
validator.validateForSender(
builder.createTransaction(senderKeyPair), null, OptionalLong.of(10)))
assertThat(validator.validateForSender(builder.createTransaction(senderKeyPair), null, false))
.isEqualTo(ValidationResult.valid());
}

Expand Down