From 19460d4fc0df6d43139b299d64cd51fbb1536dca Mon Sep 17 00:00:00 2001 From: Philipp Gackstatter Date: Tue, 19 Mar 2024 07:58:46 +0100 Subject: [PATCH 1/3] Add invalid signature mempool test in docker --- tools/docker-network/.env | 1 + .../tests/mempool_invalid_signatures_test.go | 74 +++++++++++++++++++ tools/docker-network/tests/utils.go | 34 +++++++++ 3 files changed, 109 insertions(+) create mode 100644 tools/docker-network/tests/mempool_invalid_signatures_test.go diff --git a/tools/docker-network/.env b/tools/docker-network/.env index efb1a5b3e..b3450c1aa 100644 --- a/tools/docker-network/.env +++ b/tools/docker-network/.env @@ -2,6 +2,7 @@ COMMON_CONFIG=" -c config.json --logger.level=debug --logger.outputPaths=stdout +--retainer.debugStoreErrorMessages=true --p2p.db.path=/app/data/peerdb --profiling.enabled=true --profiling.bindAddress=0.0.0.0:6061 diff --git a/tools/docker-network/tests/mempool_invalid_signatures_test.go b/tools/docker-network/tests/mempool_invalid_signatures_test.go new file mode 100644 index 000000000..75f897fa7 --- /dev/null +++ b/tools/docker-network/tests/mempool_invalid_signatures_test.go @@ -0,0 +1,74 @@ +//go:build dockertests + +package tests + +import ( + "context" + "fmt" + "testing" + "time" + + "github.com/stretchr/testify/require" + + iotago "github.com/iotaledger/iota.go/v4" + "github.com/iotaledger/iota.go/v4/api" +) + +func Test_MempoolInvalidSignatures(t *testing.T) { + d := NewDockerTestFramework(t, + WithProtocolParametersOptions( + iotago.WithTimeProviderOptions(5, time.Now().Unix(), 10, 4), + iotago.WithLivenessOptions(10, 10, 2, 4, 8), + iotago.WithRewardsOptions(8, 10, 2, 384), + iotago.WithTargetCommitteeSize(4), + )) + defer d.Stop() + + d.AddValidatorNode("V1", "docker-network-inx-validator-1-1", "http://localhost:8050", "rms1pzg8cqhfxqhq7pt37y8cs4v5u4kcc48lquy2k73ehsdhf5ukhya3y5rx2w6") + d.AddValidatorNode("V2", "docker-network-inx-validator-2-1", "http://localhost:8060", "rms1pqm4xk8e9ny5w5rxjkvtp249tfhlwvcshyr3pc0665jvp7g3hc875k538hl") + d.AddValidatorNode("V3", "docker-network-inx-validator-3-1", "http://localhost:8070", "rms1pp4wuuz0y42caz48vv876qfpmffswsvg40zz8v79sy8cp0jfxm4kunflcgt") + d.AddValidatorNode("V4", "docker-network-inx-validator-4-1", "http://localhost:8040", "rms1pr8cxs3dzu9xh4cduff4dd4cxdthpjkpwmz2244f75m0urslrsvtsshrrjw") + d.AddNode("node5", "docker-network-node-5-1", "http://localhost:8090") + + err := d.Run() + require.NoError(t, err) + + d.WaitUntilNetworkReady() + + account := d.CreateAccount() + + ctx := context.Background() + fundsOutputID := d.RequestFaucetFunds(ctx, iotago.AddressEd25519) + input := d.wallet.Output(fundsOutputID) + validTX := d.wallet.CreateBasicOutputFromInput(input, account.ID) + invalidTX := validTX.Clone().(*iotago.SignedTransaction) + + // Make validTX invalid by replacing the first unlock with an empty signature unlock. + _, is := invalidTX.Unlocks[0].(*iotago.SignatureUnlock) + if !is { + panic("expected signature unlock as first unlock") + } + invalidTX.Unlocks[0] = &iotago.SignatureUnlock{ + Signature: &iotago.Ed25519Signature{}, + } + + issuerResp, congestionResp := d.PrepareBlockIssuance(ctx, d.wallet.DefaultClient(), account.ID.ToAddress().(*iotago.AccountAddress)) + block1 := d.CreateBlock(invalidTX, account.ID, congestionResp, issuerResp) + fmt.Println("Submitting block with invalid TX") + d.SubmitBlock(context.Background(), block1) + + d.AwaitTransactionState(ctx, invalidTX.Transaction.MustID(), api.TransactionStateFailed) + d.AwaitTransactionFailure(ctx, invalidTX.Transaction.MustID(), api.TxFailureUnlockSignatureInvalid) + + issuerResp, congestionResp = d.PrepareBlockIssuance(ctx, d.wallet.DefaultClient(), account.ID.ToAddress().(*iotago.AccountAddress)) + block2 := d.CreateBlock(validTX, account.ID, congestionResp, issuerResp) + fmt.Println("Submitting block with valid TX") + d.SubmitBlock(context.Background(), block2) + + issuerResp, congestionResp = d.PrepareBlockIssuance(ctx, d.wallet.DefaultClient(), account.ID.ToAddress().(*iotago.AccountAddress)) + block3 := d.CreateBlock(invalidTX, account.ID, congestionResp, issuerResp) + fmt.Println("Submitting block with invalid TX (again)") + d.SubmitBlock(context.Background(), block3) + + d.AwaitTransactionPayloadAccepted(ctx, validTX.Transaction.MustID()) +} diff --git a/tools/docker-network/tests/utils.go b/tools/docker-network/tests/utils.go index f42bee468..10d922574 100644 --- a/tools/docker-network/tests/utils.go +++ b/tools/docker-network/tests/utils.go @@ -214,6 +214,40 @@ func (d *DockerTestFramework) AwaitTransactionPayloadAccepted(ctx context.Contex }) } +func (d *DockerTestFramework) AwaitTransactionState(ctx context.Context, txID iotago.TransactionID, expectedState api.TransactionState) { + clt := d.wallet.DefaultClient() + + d.Eventually(func() error { + resp, err := clt.TransactionMetadata(ctx, txID) + if err != nil { + return err + } + + if expectedState == resp.TransactionState { + return nil + } else { + return ierrors.Errorf("expected transaction %s to have state %s, got %s instead", txID, expectedState, resp.TransactionState) + } + }) +} + +func (d *DockerTestFramework) AwaitTransactionFailure(ctx context.Context, txID iotago.TransactionID, expectedReason api.TransactionFailureReason) { + clt := d.wallet.DefaultClient() + + d.Eventually(func() error { + resp, err := clt.TransactionMetadata(ctx, txID) + if err != nil { + return err + } + + if expectedReason == resp.TransactionFailureReason { + return nil + } else { + return ierrors.Errorf("expected transaction %s to have failure reason %T, got %s instead, failure details: %s", txID, expectedReason, resp.TransactionState, resp.TransactionFailureDetails) + } + }) +} + func (d *DockerTestFramework) AwaitCommitment(targetSlot iotago.SlotIndex) { currentCommittedSlot := d.NodeStatus("V1").LatestCommitmentID.Slot() From 09135e0b706534bb41519b49369f25c57797070b Mon Sep 17 00:00:00 2001 From: Philipp Gackstatter Date: Tue, 19 Mar 2024 08:42:43 +0100 Subject: [PATCH 2/3] Fix typo --- pkg/tests/mempool_invalid_signatures_test.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pkg/tests/mempool_invalid_signatures_test.go b/pkg/tests/mempool_invalid_signatures_test.go index 1536237ac..18de7e71d 100644 --- a/pkg/tests/mempool_invalid_signatures_test.go +++ b/pkg/tests/mempool_invalid_signatures_test.go @@ -32,7 +32,7 @@ func Test_MempoolInvalidSignatures(t *testing.T) { validTX := wallet.CreateBasicOutputsEquallyFromInput("TX", 1, "Genesis:0") invalidTX := validTX.Clone().(*iotago.SignedTransaction) - // Make tx3 invalid by replacing the first unlock with an empty signature unlock. + // Make validTX invalid by replacing the first unlock with an empty signature unlock. _, is := invalidTX.Unlocks[0].(*iotago.SignatureUnlock) if !is { panic("expected signature unlock as first unlock") From 7e6443d5406e5b6aea43694193a27e1df1e86d09 Mon Sep 17 00:00:00 2001 From: Philipp Gackstatter Date: Tue, 19 Mar 2024 10:32:40 +0100 Subject: [PATCH 3/3] Use SubmitPayload instead of SubmitBlock --- .../tests/mempool_invalid_signatures_test.go | 19 +++++++------------ 1 file changed, 7 insertions(+), 12 deletions(-) diff --git a/tools/docker-network/tests/mempool_invalid_signatures_test.go b/tools/docker-network/tests/mempool_invalid_signatures_test.go index 75f897fa7..85112ca10 100644 --- a/tools/docker-network/tests/mempool_invalid_signatures_test.go +++ b/tools/docker-network/tests/mempool_invalid_signatures_test.go @@ -45,30 +45,25 @@ func Test_MempoolInvalidSignatures(t *testing.T) { // Make validTX invalid by replacing the first unlock with an empty signature unlock. _, is := invalidTX.Unlocks[0].(*iotago.SignatureUnlock) - if !is { - panic("expected signature unlock as first unlock") - } + require.Truef(t, is, "expected signature unlock as first unlock") invalidTX.Unlocks[0] = &iotago.SignatureUnlock{ Signature: &iotago.Ed25519Signature{}, } - issuerResp, congestionResp := d.PrepareBlockIssuance(ctx, d.wallet.DefaultClient(), account.ID.ToAddress().(*iotago.AccountAddress)) - block1 := d.CreateBlock(invalidTX, account.ID, congestionResp, issuerResp) fmt.Println("Submitting block with invalid TX") - d.SubmitBlock(context.Background(), block1) + issuerResp, congestionResp := d.PrepareBlockIssuance(ctx, d.wallet.DefaultClient(), account.ID.ToAddress().(*iotago.AccountAddress)) + d.SubmitPayload(context.Background(), invalidTX, account.ID, congestionResp, issuerResp) d.AwaitTransactionState(ctx, invalidTX.Transaction.MustID(), api.TransactionStateFailed) d.AwaitTransactionFailure(ctx, invalidTX.Transaction.MustID(), api.TxFailureUnlockSignatureInvalid) - issuerResp, congestionResp = d.PrepareBlockIssuance(ctx, d.wallet.DefaultClient(), account.ID.ToAddress().(*iotago.AccountAddress)) - block2 := d.CreateBlock(validTX, account.ID, congestionResp, issuerResp) fmt.Println("Submitting block with valid TX") - d.SubmitBlock(context.Background(), block2) - issuerResp, congestionResp = d.PrepareBlockIssuance(ctx, d.wallet.DefaultClient(), account.ID.ToAddress().(*iotago.AccountAddress)) - block3 := d.CreateBlock(invalidTX, account.ID, congestionResp, issuerResp) + d.SubmitPayload(context.Background(), validTX, account.ID, congestionResp, issuerResp) + fmt.Println("Submitting block with invalid TX (again)") - d.SubmitBlock(context.Background(), block3) + issuerResp, congestionResp = d.PrepareBlockIssuance(ctx, d.wallet.DefaultClient(), account.ID.ToAddress().(*iotago.AccountAddress)) + d.SubmitPayload(context.Background(), invalidTX, account.ID, congestionResp, issuerResp) d.AwaitTransactionPayloadAccepted(ctx, validTX.Transaction.MustID()) }