From de57ae6181fe54d402f629f5a2202be3cad6bd16 Mon Sep 17 00:00:00 2001 From: Lazar Date: Wed, 27 Nov 2024 17:05:49 +0100 Subject: [PATCH 1/5] testing 1000 unbonds in a block --- e2etest/test_manager.go | 2 + e2etest/test_manager_btcstaking.go | 5 +- e2etest/unbondingwatcher_e2e_test.go | 141 ++++++++++++++++++--------- 3 files changed, 102 insertions(+), 46 deletions(-) diff --git a/e2etest/test_manager.go b/e2etest/test_manager.go index 16df8a28..b5e68f32 100644 --- a/e2etest/test_manager.go +++ b/e2etest/test_manager.go @@ -12,6 +12,7 @@ import ( "go.uber.org/zap" "os" "path/filepath" + "sync" "testing" "time" @@ -61,6 +62,7 @@ type TestManager struct { Config *config.Config WalletPrivKey *btcec.PrivateKey manger *container.Manager + mu sync.Mutex } func initBTCClientWithSubscriber(t *testing.T, cfg *config.Config) *btcclient.Client { diff --git a/e2etest/test_manager_btcstaking.go b/e2etest/test_manager_btcstaking.go index 600611ca..76831b15 100644 --- a/e2etest/test_manager_btcstaking.go +++ b/e2etest/test_manager_btcstaking.go @@ -104,6 +104,7 @@ func (tm *TestManager) CreateBTCDelegation( require.NoError(t, err) stakingTimeBlocks := bsParams.Params.MaxStakingTimeBlocks // get top UTXO + tm.mu.Lock() topUnspentResult, _, err := tm.BTCClient.GetHighUTXOAndSum() require.NoError(t, err) topUTXO, err := types.NewUTXO(topUnspentResult, regtestParams) @@ -123,7 +124,7 @@ func (tm *TestManager) CreateBTCDelegation( }, eventuallyWaitTimeOut, eventuallyPollTime) mBlock := tm.mineBlock(t) - require.Equal(t, 2, len(mBlock.Transactions)) + //require.Equal(t, 2, len(mBlock.Transactions)) // wait until staking tx is on Bitcoin require.Eventually(t, func() bool { @@ -144,6 +145,8 @@ func (tm *TestManager) CreateBTCDelegation( stakingOutIdx, err := outIdx(stakingSlashingInfo.StakingTx, stakingSlashingInfo.StakingInfo.StakingOutput) require.NoError(t, err) + tm.mu.Unlock() + // create PoP pop, err := bstypes.NewPoPBTC(addr, tm.WalletPrivKey) require.NoError(t, err) diff --git a/e2etest/unbondingwatcher_e2e_test.go b/e2etest/unbondingwatcher_e2e_test.go index 80cbde8e..3b32f233 100644 --- a/e2etest/unbondingwatcher_e2e_test.go +++ b/e2etest/unbondingwatcher_e2e_test.go @@ -1,13 +1,13 @@ -//go:build e2e -// +build e2e - package e2etest import ( "fmt" + "github.com/babylonlabs-io/babylon/testutil/datagen" btcstakingtypes "github.com/babylonlabs-io/babylon/x/btcstaking/types" + "github.com/btcsuite/btcd/btcec/v2" promtestutil "github.com/prometheus/client_golang/prometheus/testutil" "go.uber.org/zap" + "golang.org/x/sync/errgroup" "sync" "testing" "time" @@ -26,7 +26,7 @@ import ( func TestUnbondingWatcher(t *testing.T) { t.Parallel() // segwit is activated at height 300. It's needed by staking/slashing tx - numMatureOutputs := uint32(300) + numMatureOutputs := uint32(1200) tm := StartManager(t, numMatureOutputs, defaultEpochInterval) defer tm.Stop(t) @@ -64,66 +64,116 @@ func TestUnbondingWatcher(t *testing.T) { // set up a finality provider _, fpSK := tm.CreateFinalityProvider(t) - // set up a BTC delegation - stakingSlashingInfo, unbondingSlashingInfo, delSK := tm.CreateBTCDelegation(t, fpSK) - - // Staker unbonds by directly sending tx to btc network. Watcher should detect it and report to babylon. - unbondingPathSpendInfo, err := stakingSlashingInfo.StakingInfo.UnbondingPathSpendInfo() - require.NoError(t, err) - stakingOutIdx, err := outIdx(unbondingSlashingInfo.UnbondingTx, unbondingSlashingInfo.UnbondingInfo.UnbondingOutput) - require.NoError(t, err) - - unbondingTxSchnorrSig, err := btcstaking.SignTxWithOneScriptSpendInputStrict( - unbondingSlashingInfo.UnbondingTx, - stakingSlashingInfo.StakingTx, - stakingOutIdx, - unbondingPathSpendInfo.GetPkScriptPath(), - delSK, - ) - require.NoError(t, err) - resp, err := tm.BabylonClient.BTCDelegation(stakingSlashingInfo.StakingTx.TxHash().String()) - require.NoError(t, err) + type testStakingBundle struct { + stakingSlashingInfo *datagen.TestStakingSlashingInfo + unbondingSlashingInfo *datagen.TestUnbondingSlashingInfo + delSK *btcec.PrivateKey + } - covenantSigs := resp.BtcDelegation.UndelegationResponse.CovenantUnbondingSigList - witness, err := unbondingPathSpendInfo.CreateUnbondingPathWitness( - []*schnorr.Signature{covenantSigs[0].Sig.MustToBTCSig()}, - unbondingTxSchnorrSig, - ) - unbondingSlashingInfo.UnbondingTx.TxIn[0].Witness = witness + var delegations []testStakingBundle + expectedUnbonds := 1000 + + var gr errgroup.Group + for i := 0; i < expectedUnbonds; i++ { + gr.Go(func() error { + // set up a BTC delegation + stakingSlashingInfo, unbondingSlashingInfo, delSK := tm.CreateBTCDelegation(t, fpSK) + delegations = append(delegations, testStakingBundle{ + stakingSlashingInfo: stakingSlashingInfo, + unbondingSlashingInfo: unbondingSlashingInfo, + delSK: delSK, + }) + + return nil + }) + } + _ = gr.Wait() - // Send unbonding tx to Bitcoin - _, err = tm.BTCClient.SendRawTransaction(unbondingSlashingInfo.UnbondingTx, true) - require.NoError(t, err) + var wg sync.WaitGroup + wg.Add(expectedUnbonds) + for _, del := range delegations { + go func() { + defer wg.Done() + stakingSlashingInfo := del.stakingSlashingInfo + unbondingSlashingInfo := del.unbondingSlashingInfo + delSK := del.delSK + + // Staker unbonds by directly sending tx to btc network. Watcher should detect it and report to babylon. + unbondingPathSpendInfo, err := stakingSlashingInfo.StakingInfo.UnbondingPathSpendInfo() + require.NoError(t, err) + stakingOutIdx, err := outIdx(unbondingSlashingInfo.UnbondingTx, unbondingSlashingInfo.UnbondingInfo.UnbondingOutput) + require.NoError(t, err) + + unbondingTxSchnorrSig, err := btcstaking.SignTxWithOneScriptSpendInputStrict( + unbondingSlashingInfo.UnbondingTx, + stakingSlashingInfo.StakingTx, + stakingOutIdx, + unbondingPathSpendInfo.GetPkScriptPath(), + delSK, + ) + require.NoError(t, err) + + resp, err := tm.BabylonClient.BTCDelegation(stakingSlashingInfo.StakingTx.TxHash().String()) + require.NoError(t, err) + + covenantSigs := resp.BtcDelegation.UndelegationResponse.CovenantUnbondingSigList + witness, err := unbondingPathSpendInfo.CreateUnbondingPathWitness( + []*schnorr.Signature{covenantSigs[0].Sig.MustToBTCSig()}, + unbondingTxSchnorrSig, + ) + unbondingSlashingInfo.UnbondingTx.TxIn[0].Witness = witness + + // Send unbonding tx to Bitcoin + _, err = tm.BTCClient.SendRawTransaction(unbondingSlashingInfo.UnbondingTx, true) + require.NoError(t, err) + + // mine a block with this tx, and insert it to Bitcoin + unbondingTxHash := unbondingSlashingInfo.UnbondingTx.TxHash() + t.Logf("submitted unbonding tx with hash %s", unbondingTxHash.String()) + require.Eventually(t, func() bool { + return len(tm.RetrieveTransactionFromMempool(t, []*chainhash.Hash{&unbondingTxHash})) == 1 + }, eventuallyWaitTimeOut, eventuallyPollTime) + }() + } - // mine a block with this tx, and insert it to Bitcoin - unbondingTxHash := unbondingSlashingInfo.UnbondingTx.TxHash() - t.Logf("submitted unbonding tx with hash %s", unbondingTxHash.String()) - require.Eventually(t, func() bool { - return len(tm.RetrieveTransactionFromMempool(t, []*chainhash.Hash{&unbondingTxHash})) == 1 - }, eventuallyWaitTimeOut, eventuallyPollTime) + wg.Wait() minedBlock := tm.mineBlock(t) - require.Equal(t, 2, len(minedBlock.Transactions)) + require.Equal(t, expectedUnbonds+1, len(minedBlock.Transactions)) tm.CatchUpBTCLightClient(t) + unbonded := make(map[string]bool) + t0 := time.Now() require.Eventually(t, func() bool { - resp, err := tm.BabylonClient.BTCDelegation(stakingSlashingInfo.StakingTx.TxHash().String()) - require.NoError(t, err) + for _, ssi := range delegations { + resp, err := tm.BabylonClient.BTCDelegation(ssi.stakingSlashingInfo.StakingTx.TxHash().String()) + require.NoError(t, err) - // TODO: Add field for staker signature in BTCDelegation query to check it directly, - // for now it is enough to check that delegation is not active, as if unbonding was reported - // delegation will be deactivated - return !resp.BtcDelegation.Active + if !resp.BtcDelegation.Active { + unbonded[ssi.stakingSlashingInfo.StakingTx.TxHash().String()] = true + } + } + + countActive := 0 + for _, unb := range unbonded { + if !unb { + return false + } + countActive++ + } + return countActive == len(delegations) }, eventuallyWaitTimeOut, eventuallyPollTime) + t.Logf("time took for all delegations to be unbonded %s", time.Since(t0)) } // TestActivatingDelegation verifies that a delegation created without an inclusion proof will // eventually become "active". // Specifically, that stakingEventWatcher will send a MsgAddBTCDelegationInclusionProof to do so. func TestActivatingDelegation(t *testing.T) { + t.Skip() t.Parallel() // segwit is activated at height 300. It's necessary for staking/slashing tx numMatureOutputs := uint32(300) @@ -226,6 +276,7 @@ func TestActivatingDelegation(t *testing.T) { // In this test, we include both staking tx and unbonding tx in the same block. // The delegation goes through "VERIFIED" → "ACTIVE" → "UNBONDED" status throughout this test. func TestActivatingAndUnbondingDelegation(t *testing.T) { + t.Skip() //t.Parallel() // segwit is activated at height 300. It's necessary for staking/slashing tx numMatureOutputs := uint32(300) From 9eafa859b9018e82a014fc5d97e07c2a772d2d5e Mon Sep 17 00:00:00 2001 From: Lazar Date: Wed, 27 Nov 2024 17:09:44 +0100 Subject: [PATCH 2/5] timeout and build flags --- Makefile | 2 +- e2etest/unbondingwatcher_e2e_test.go | 3 +++ 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/Makefile b/Makefile index a519fb29..cf6cd606 100644 --- a/Makefile +++ b/Makefile @@ -48,7 +48,7 @@ test: go test -race ./... test-e2e: - go test -mod=readonly -failfast -timeout=15m -v $(PACKAGES_E2E) -count=1 --parallel 12 --tags=e2e + go test -mod=readonly -failfast -timeout=1005m -v $(PACKAGES_E2E) -count=1 --parallel 12 --tags=e2e build-docker: $(DOCKER) build --tag babylonlabs-io/vigilante -f Dockerfile \ diff --git a/e2etest/unbondingwatcher_e2e_test.go b/e2etest/unbondingwatcher_e2e_test.go index 3b32f233..9df557be 100644 --- a/e2etest/unbondingwatcher_e2e_test.go +++ b/e2etest/unbondingwatcher_e2e_test.go @@ -1,3 +1,6 @@ +//go:build e2e +// +build e2e + package e2etest import ( From c4a6297979dd1ff10e744839d345e1ead40bdac6 Mon Sep 17 00:00:00 2001 From: Lazar Date: Wed, 27 Nov 2024 17:13:54 +0100 Subject: [PATCH 3/5] wait --- e2etest/unbondingwatcher_e2e_test.go | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/e2etest/unbondingwatcher_e2e_test.go b/e2etest/unbondingwatcher_e2e_test.go index 9df557be..363691ef 100644 --- a/e2etest/unbondingwatcher_e2e_test.go +++ b/e2etest/unbondingwatcher_e2e_test.go @@ -90,6 +90,10 @@ func TestUnbondingWatcher(t *testing.T) { return nil }) + // wait for a bit to combat contention + if (i+1)%100 == 0 { + time.Sleep(7 * time.Second) + } } _ = gr.Wait() From 77abb1dfa3adf59575d6048b0e499e91806a0031 Mon Sep 17 00:00:00 2001 From: Lazar Date: Wed, 27 Nov 2024 17:33:32 +0100 Subject: [PATCH 4/5] long unlock --- e2etest/test_manager.go | 2 +- e2etest/unbondingwatcher_e2e_test.go | 5 ++++- 2 files changed, 5 insertions(+), 2 deletions(-) diff --git a/e2etest/test_manager.go b/e2etest/test_manager.go index b5e68f32..70d3d309 100644 --- a/e2etest/test_manager.go +++ b/e2etest/test_manager.go @@ -109,7 +109,7 @@ func StartManager(t *testing.T, numMatureOutputsInWallet uint32, epochInterval u }, nil) require.NoError(t, err) - err = testRpcClient.WalletPassphrase(passphrase, 600) + err = testRpcClient.WalletPassphrase(passphrase, 6000) require.NoError(t, err) walletPrivKey, err := importPrivateKey(btcHandler) diff --git a/e2etest/unbondingwatcher_e2e_test.go b/e2etest/unbondingwatcher_e2e_test.go index 363691ef..4a88d7fa 100644 --- a/e2etest/unbondingwatcher_e2e_test.go +++ b/e2etest/unbondingwatcher_e2e_test.go @@ -99,7 +99,7 @@ func TestUnbondingWatcher(t *testing.T) { var wg sync.WaitGroup wg.Add(expectedUnbonds) - for _, del := range delegations { + for i, del := range delegations { go func() { defer wg.Done() stakingSlashingInfo := del.stakingSlashingInfo @@ -142,6 +142,9 @@ func TestUnbondingWatcher(t *testing.T) { return len(tm.RetrieveTransactionFromMempool(t, []*chainhash.Hash{&unbondingTxHash})) == 1 }, eventuallyWaitTimeOut, eventuallyPollTime) }() + if (i+1)%100 == 0 { + time.Sleep(7 * time.Second) + } } wg.Wait() From 88fc3cb2022ccb830907a6c764df816f411965bc Mon Sep 17 00:00:00 2001 From: Lazar Date: Wed, 27 Nov 2024 19:21:27 +0100 Subject: [PATCH 5/5] long wait, slow poll --- e2etest/unbondingwatcher_e2e_test.go | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/e2etest/unbondingwatcher_e2e_test.go b/e2etest/unbondingwatcher_e2e_test.go index 4a88d7fa..0439bac4 100644 --- a/e2etest/unbondingwatcher_e2e_test.go +++ b/e2etest/unbondingwatcher_e2e_test.go @@ -50,7 +50,7 @@ func TestUnbondingWatcher(t *testing.T) { commonCfg := config.DefaultCommonConfig() bstCfg := config.DefaultBTCStakingTrackerConfig() - bstCfg.CheckDelegationsInterval = 1 * time.Second + bstCfg.CheckDelegationsInterval = 2 * time.Second stakingTrackerMetrics := metrics.NewBTCStakingTrackerMetrics() bsTracker := bst.NewBTCStakingTracker( @@ -75,7 +75,7 @@ func TestUnbondingWatcher(t *testing.T) { } var delegations []testStakingBundle - expectedUnbonds := 1000 + expectedUnbonds := 800 var gr errgroup.Group for i := 0; i < expectedUnbonds; i++ { @@ -175,7 +175,7 @@ func TestUnbondingWatcher(t *testing.T) { } return countActive == len(delegations) - }, eventuallyWaitTimeOut, eventuallyPollTime) + }, 15*time.Minute, 2*time.Second) t.Logf("time took for all delegations to be unbonded %s", time.Since(t0)) }