From 99dab44c20be7009bc3357d8af846ac06994e6ae Mon Sep 17 00:00:00 2001 From: muXxer Date: Fri, 17 May 2024 17:17:31 +0200 Subject: [PATCH 1/3] Fix potential underflow in genesis snapshot generation --- pkg/testsuite/snapshotcreator/snapshotcreator.go | 13 ++++++++++++- 1 file changed, 12 insertions(+), 1 deletion(-) diff --git a/pkg/testsuite/snapshotcreator/snapshotcreator.go b/pkg/testsuite/snapshotcreator/snapshotcreator.go index fcc38bf53..e117a426e 100644 --- a/pkg/testsuite/snapshotcreator/snapshotcreator.go +++ b/pkg/testsuite/snapshotcreator/snapshotcreator.go @@ -3,6 +3,7 @@ package snapshotcreator import ( "os" + "github.com/iotaledger/hive.go/core/safemath" "github.com/iotaledger/hive.go/ierrors" "github.com/iotaledger/hive.go/lo" "github.com/iotaledger/hive.go/log" @@ -137,8 +138,18 @@ func CreateSnapshot(opts ...options.Option[Options]) error { return accumulator + details.Amount }, iotago.BaseToken(0)) + supplyMinusAccounts, err := safemath.SafeSub(opt.ProtocolParameters.TokenSupply(), totalAccountAmount) + if err != nil { + return ierrors.Wrapf(err, "failed to calculate genesis output balance, remaining funds: %d, needed account funds: %d", opt.ProtocolParameters.TokenSupply(), totalAccountAmount) + } + + genesisOutputBalance, err := safemath.SafeSub(supplyMinusAccounts, totalBasicOutputAmount) + if err != nil { + return ierrors.Wrapf(err, "failed to calculate genesis output balance, remaining funds: %d, needed basic output funds: %d", supplyMinusAccounts, totalBasicOutputAmount) + } + var genesisTransactionOutputs iotago.TxEssenceOutputs - genesisOutput, err := createGenesisOutput(api, opt.ProtocolParameters.TokenSupply()-totalAccountAmount-totalBasicOutputAmount, iotago.MaxMana/100, opt.GenesisKeyManager) + genesisOutput, err := createGenesisOutput(api, genesisOutputBalance, iotago.MaxMana/100, opt.GenesisKeyManager) if err != nil { return ierrors.Wrap(err, "failed to create genesis outputs") } From e933ffc48e545316f5d26d01ee49317141a1e8bb Mon Sep 17 00:00:00 2001 From: muXxer Date: Fri, 17 May 2024 18:38:07 +0200 Subject: [PATCH 2/3] Fix Test_Validator_PerfectIssuanceWithHugeStake --- pkg/tests/validator_test.go | 19 ++++++++++++++----- 1 file changed, 14 insertions(+), 5 deletions(-) diff --git a/pkg/tests/validator_test.go b/pkg/tests/validator_test.go index 699b1ba8b..fd4750051 100644 --- a/pkg/tests/validator_test.go +++ b/pkg/tests/validator_test.go @@ -21,7 +21,7 @@ import ( // Supply for the test with faster slot duration and slots per epoch. const SUPPLY = iotago.BaseToken(1_813_620_509_061_365) -func setupValidatorTestsuite(t *testing.T, walletOpts ...options.Option[testsuite.WalletOptions]) *testsuite.TestSuite { +func setupValidatorTestsuiteWithStake(t *testing.T, validator1Balance iotago.BaseToken, validator2Balance iotago.BaseToken, walletOpts ...options.Option[testsuite.WalletOptions]) *testsuite.TestSuite { var slotDuration uint8 = 5 var slotsPerEpochExponent uint8 = 5 var validationBlocksPerSlot uint8 = 5 @@ -54,13 +54,13 @@ func setupValidatorTestsuite(t *testing.T, walletOpts ...options.Option[testsuit // Add validator nodes to the network. This will add validator accounts to the snapshot. vnode1 := ts.AddValidatorNode("node1", append( []options.Option[testsuite.WalletOptions]{ - testsuite.WithWalletAmount(20_000_000), + testsuite.WithWalletAmount(validator1Balance), }, walletOpts..., )...) vnode2 := ts.AddValidatorNode("node2", append( []options.Option[testsuite.WalletOptions]{ - testsuite.WithWalletAmount(25_000_000), + testsuite.WithWalletAmount(validator2Balance), }, walletOpts..., )...) @@ -82,6 +82,10 @@ func setupValidatorTestsuite(t *testing.T, walletOpts ...options.Option[testsuit return ts } +func setupValidatorTestsuite(t *testing.T, walletOpts ...options.Option[testsuite.WalletOptions]) *testsuite.TestSuite { + return setupValidatorTestsuiteWithStake(t, 20_000_000, 25_000_000, walletOpts...) +} + type EpochPerformanceMap = map[iotago.EpochIndex]uint64 type ValidatorTest struct { ts *testsuite.TestSuite @@ -137,9 +141,14 @@ func Test_Validator_PerfectIssuanceWithNonZeroFixedCost(t *testing.T) { } func Test_Validator_PerfectIssuanceWithHugeStake(t *testing.T) { - // This gives both validators the max supply as stake, which is unrealistic, + // This gives one validator almost the max supply as stake, which is unrealistic, // but is supposed to test if one validator with a huge stake causes an overflow in the rewards calculation. - ts := setupValidatorTestsuite(t, testsuite.WithWalletAmount(SUPPLY)) + var validator1Balance iotago.BaseToken = 25_000_000 + var otherAccountBalance iotago.BaseToken = 31_700 + var genesisOutputBalance iotago.BaseToken = 14_100 + var validator2Balance iotago.BaseToken = SUPPLY - validator1Balance - otherAccountBalance - genesisOutputBalance + + ts := setupValidatorTestsuiteWithStake(t, iotago.BaseToken(validator1Balance), validator2Balance) defer ts.Shutdown() validationBlocksPerSlot := ts.API.ProtocolParameters().ValidationBlocksPerSlot() From ce479a89f9ebeda8fbe0bcaeedf266e195d50cb0 Mon Sep 17 00:00:00 2001 From: muXxer Date: Fri, 17 May 2024 18:38:38 +0200 Subject: [PATCH 3/3] Fix nil pointer genesis output if no balance left --- pkg/testsuite/snapshotcreator/snapshotcreator.go | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/pkg/testsuite/snapshotcreator/snapshotcreator.go b/pkg/testsuite/snapshotcreator/snapshotcreator.go index e117a426e..a7926cd55 100644 --- a/pkg/testsuite/snapshotcreator/snapshotcreator.go +++ b/pkg/testsuite/snapshotcreator/snapshotcreator.go @@ -153,7 +153,9 @@ func CreateSnapshot(opts ...options.Option[Options]) error { if err != nil { return ierrors.Wrap(err, "failed to create genesis outputs") } - genesisTransactionOutputs = append(genesisTransactionOutputs, genesisOutput) + if genesisOutput != nil { + genesisTransactionOutputs = append(genesisTransactionOutputs, genesisOutput) + } accountOutputs, err := createGenesisAccounts(api, opt.Accounts) if err != nil {