From 8b853879752a8253c7681387b5fd96b8d3b734f5 Mon Sep 17 00:00:00 2001 From: lazar Date: Fri, 4 Oct 2024 14:05:23 +0200 Subject: [PATCH 1/7] e2e for activating and unbonding --- .../stakingeventwatcher.go | 23 ++- .../tracked_delegations.go | 48 +++++- e2etest/test_manager_btcstaking.go | 31 +--- e2etest/unbondingwatcher_e2e_test.go | 146 +++++++++++++++++- 4 files changed, 202 insertions(+), 46 deletions(-) diff --git a/btcstaking-tracker/stakingeventwatcher/stakingeventwatcher.go b/btcstaking-tracker/stakingeventwatcher/stakingeventwatcher.go index 62a9a36..a95a3d0 100644 --- a/btcstaking-tracker/stakingeventwatcher/stakingeventwatcher.go +++ b/btcstaking-tracker/stakingeventwatcher/stakingeventwatcher.go @@ -207,7 +207,7 @@ func (sew *StakingEventWatcher) fetchDelegations() { continue } - addToUnbonding := func(delegation Delegation) { + addToUnbondingFunc := func(delegation Delegation) { del := &newDelegation{ stakingTxHash: delegation.StakingTx.TxHash(), stakingTx: delegation.StakingTx, @@ -216,14 +216,19 @@ func (sew *StakingEventWatcher) fetchDelegations() { unbondingOutput: delegation.UnbondingOutput, } - // if we already have this delegation, skip it + // if we already have this delegation, we still want to check if it has changed, // we should track both verified and active status for unbonding - if sew.unbondingTracker.GetDelegation(delegation.StakingTx.TxHash()) == nil { + changed, exists := sew.unbondingTracker.HasDelegationChanged(delegation.StakingTx.TxHash(), del) + if exists && changed { + // Delegation exists and has changed, push the update. + utils.PushOrQuit(sew.unbondingDelegationChan, del, sew.quit) + } else if !exists { + // Delegation doesn't exist, push the new delegation. utils.PushOrQuit(sew.unbondingDelegationChan, del, sew.quit) } } - addToPending := func(delegation Delegation) { + addToPendingFunc := func(delegation Delegation) { del := &newDelegation{ stakingTxHash: delegation.StakingTx.TxHash(), stakingTx: delegation.StakingTx, @@ -237,6 +242,8 @@ func (sew *StakingEventWatcher) fetchDelegations() { del.stakingTx, del.stakingOutputIdx, del.unbondingOutput, + del.delegationStartHeight, + false, ) } } @@ -246,21 +253,21 @@ func (sew *StakingEventWatcher) fetchDelegations() { go func() { defer wg.Done() - if err = sew.checkBabylonDelegations(btcstakingtypes.BTCDelegationStatus_ACTIVE, addToUnbonding); err != nil { + if err = sew.checkBabylonDelegations(btcstakingtypes.BTCDelegationStatus_ACTIVE, addToUnbondingFunc); err != nil { sew.logger.Errorf("error checking babylon delegations: %v", err) } }() go func() { defer wg.Done() - if err = sew.checkBabylonDelegations(btcstakingtypes.BTCDelegationStatus_VERIFIED, addToUnbonding); err != nil { + if err = sew.checkBabylonDelegations(btcstakingtypes.BTCDelegationStatus_VERIFIED, addToUnbondingFunc); err != nil { sew.logger.Errorf("error checking babylon delegations: %v", err) } }() go func() { defer wg.Done() - if err = sew.checkBabylonDelegations(btcstakingtypes.BTCDelegationStatus_VERIFIED, addToPending); err != nil { + if err = sew.checkBabylonDelegations(btcstakingtypes.BTCDelegationStatus_VERIFIED, addToPendingFunc); err != nil { sew.logger.Errorf("error checking babylon delegations: %v", err) } }() @@ -457,6 +464,8 @@ func (sew *StakingEventWatcher) handleUnbondedDelegations() { activeDel.stakingTx, activeDel.stakingOutputIdx, activeDel.unbondingOutput, + activeDel.delegationStartHeight, + true, ) if err != nil { diff --git a/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go b/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go index 293db8b..63ec7a2 100644 --- a/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go +++ b/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go @@ -2,6 +2,7 @@ package stakingeventwatcher import ( "fmt" + "reflect" "sync" "github.com/btcsuite/btcd/chaincfg/chainhash" @@ -9,9 +10,10 @@ import ( ) type TrackedDelegation struct { - StakingTx *wire.MsgTx - StakingOutputIdx uint32 - UnbondingOutput *wire.TxOut + StakingTx *wire.MsgTx + StakingOutputIdx uint32 + UnbondingOutput *wire.TxOut + DelegationStartHeight uint64 } type TrackedDelegations struct { @@ -60,11 +62,14 @@ func (td *TrackedDelegations) AddDelegation( StakingTx *wire.MsgTx, StakingOutputIdx uint32, UnbondingOutput *wire.TxOut, + delegationStartHeight uint64, + shouldUpdate bool, ) (*TrackedDelegation, error) { delegation := &TrackedDelegation{ - StakingTx: StakingTx, - StakingOutputIdx: StakingOutputIdx, - UnbondingOutput: UnbondingOutput, + StakingTx: StakingTx, + StakingOutputIdx: StakingOutputIdx, + UnbondingOutput: UnbondingOutput, + DelegationStartHeight: delegationStartHeight, } stakingTxHash := StakingTx.TxHash() @@ -73,6 +78,11 @@ func (td *TrackedDelegations) AddDelegation( defer td.mu.Unlock() if _, ok := td.mapping[stakingTxHash]; ok { + if shouldUpdate { + // Update the existing delegation + td.mapping[stakingTxHash] = delegation + return delegation, nil + } return nil, fmt.Errorf("delegation already tracked for staking tx hash %s", stakingTxHash) } @@ -86,3 +96,29 @@ func (td *TrackedDelegations) RemoveDelegation(stakingTxHash chainhash.Hash) { delete(td.mapping, stakingTxHash) } + +func (td *TrackedDelegations) HasDelegationChanged( + stakingTxHash chainhash.Hash, + newDelegation *newDelegation, +) (bool, bool) { + td.mu.Lock() + defer td.mu.Unlock() + + // Check if the delegation exists in the map + existingDelegation, exists := td.mapping[stakingTxHash] + if !exists { + // If it doesn't exist, return false for changed, and false for exists + return false, false + } + + // Compare fields to check if the delegation has changed + if existingDelegation.StakingOutputIdx != newDelegation.stakingOutputIdx || + existingDelegation.StakingTx.TxHash() != newDelegation.stakingTx.TxHash() || + !reflect.DeepEqual(existingDelegation.UnbondingOutput, newDelegation.unbondingOutput) || + existingDelegation.DelegationStartHeight != newDelegation.delegationStartHeight { + return true, true // The delegation has changed and it exists + } + + // The delegation exists but hasn't changed + return false, true +} diff --git a/e2etest/test_manager_btcstaking.go b/e2etest/test_manager_btcstaking.go index cc0319f..aeb4c71 100644 --- a/e2etest/test_manager_btcstaking.go +++ b/e2etest/test_manager_btcstaking.go @@ -214,7 +214,7 @@ func (tm *TestManager) CreateBTCDelegation( func (tm *TestManager) CreateBTCDelegationWithoutIncl( t *testing.T, fpSK *btcec.PrivateKey, -) (*datagen.TestStakingSlashingInfo, *datagen.TestUnbondingSlashingInfo, *btcec.PrivateKey) { +) (*wire.MsgTx, *datagen.TestStakingSlashingInfo, *datagen.TestUnbondingSlashingInfo, *btcec.PrivateKey) { signerAddr := tm.BabylonClient.MustGetAddr() addr := sdk.MustAccAddressFromBech32(signerAddr) @@ -311,34 +311,7 @@ func (tm *TestManager) CreateBTCDelegationWithoutIncl( stakingOutIdx, ) - // send staking tx to Bitcoin node's mempool - _, err = tm.BTCClient.SendRawTransaction(stakingMsgTx, true) - require.NoError(t, err) - - require.Eventually(t, func() bool { - return len(tm.RetrieveTransactionFromMempool(t, []*chainhash.Hash{stakingMsgTxHash})) == 1 - }, eventuallyWaitTimeOut, eventuallyPollTime) - - mBlock := tm.mineBlock(t) - require.Equal(t, 2, len(mBlock.Transactions)) - - // wait until staking tx is on Bitcoin - require.Eventually(t, func() bool { - _, err := tm.BTCClient.GetRawTransaction(stakingMsgTxHash) - return err == nil - }, eventuallyWaitTimeOut, eventuallyPollTime) - - // insert k empty blocks to Bitcoin - btccParamsResp, err := tm.BabylonClient.BTCCheckpointParams() - require.NoError(t, err) - btccParams := btccParamsResp.Params - for i := 0; i < int(btccParams.BtcConfirmationDepth); i++ { - tm.mineBlock(t) - } - - tm.CatchUpBTCLightClient(t) - - return stakingSlashingInfo, unbondingSlashingInfo, tm.WalletPrivKey + return stakingMsgTx, stakingSlashingInfo, unbondingSlashingInfo, tm.WalletPrivKey } func (tm *TestManager) createStakingAndSlashingTx( diff --git a/e2etest/unbondingwatcher_e2e_test.go b/e2etest/unbondingwatcher_e2e_test.go index 4d50811..94d3845 100644 --- a/e2etest/unbondingwatcher_e2e_test.go +++ b/e2etest/unbondingwatcher_e2e_test.go @@ -1,9 +1,7 @@ -//go:build e2e -// +build e2e - package e2etest import ( + btcstakingtypes "github.com/babylonlabs-io/babylon/x/btcstaking/types" "go.uber.org/zap" "testing" "time" @@ -157,7 +155,35 @@ func TestActivatingDelegation(t *testing.T) { // set up a finality provider _, fpSK := tm.CreateFinalityProvider(t) // set up a BTC delegation - stakingSlashingInfo, _, _ := tm.CreateBTCDelegationWithoutIncl(t, fpSK) + stakingMsgTx, stakingSlashingInfo, _, _ := tm.CreateBTCDelegationWithoutIncl(t, fpSK) + stakingMsgTxHash := stakingMsgTx.TxHash() + + // send staking tx to Bitcoin node's mempool + _, err = tm.BTCClient.SendRawTransaction(stakingMsgTx, true) + require.NoError(t, err) + + require.Eventually(t, func() bool { + return len(tm.RetrieveTransactionFromMempool(t, []*chainhash.Hash{&stakingMsgTxHash})) == 1 + }, eventuallyWaitTimeOut, eventuallyPollTime) + + mBlock := tm.mineBlock(t) + require.Equal(t, 2, len(mBlock.Transactions)) + + // wait until staking tx is on Bitcoin + require.Eventually(t, func() bool { + _, err := tm.BTCClient.GetRawTransaction(&stakingMsgTxHash) + return err == nil + }, eventuallyWaitTimeOut, eventuallyPollTime) + + // insert k empty blocks to Bitcoin + btccParamsResp, err := tm.BabylonClient.BTCCheckpointParams() + require.NoError(t, err) + btccParams := btccParamsResp.Params + for i := 0; i < int(btccParams.BtcConfirmationDepth); i++ { + tm.mineBlock(t) + } + + tm.CatchUpBTCLightClient(t) // created delegation lacks inclusion proof, once created it will be in // pending status, once convenant signatures are added it will be in verified status, @@ -170,3 +196,115 @@ func TestActivatingDelegation(t *testing.T) { return resp.BtcDelegation.Active }, eventuallyWaitTimeOut, eventuallyPollTime) } + +// TestActivatingAndUnbondingDelegation tests that delegation will eventually become UNBONDED given that +// both staking and unbonding tx are in the same block. In this test +func TestActivatingAndUnbondingDelegation(t *testing.T) { + t.Parallel() + // segwit is activated at height 300. It's necessary for staking/slashing tx + numMatureOutputs := uint32(300) + + tm := StartManager(t, numMatureOutputs, defaultEpochInterval) + defer tm.Stop(t) + // Insert all existing BTC headers to babylon node + tm.CatchUpBTCLightClient(t) + + btcNotifier, err := btcclient.NewNodeBackend( + btcclient.ToBitcoindConfig(tm.Config.BTC), + &chaincfg.RegressionNetParams, + &btcclient.EmptyHintCache{}, + ) + require.NoError(t, err) + + err = btcNotifier.Start() + require.NoError(t, err) + + commonCfg := config.DefaultCommonConfig() + bstCfg := config.DefaultBTCStakingTrackerConfig() + bstCfg.CheckDelegationsInterval = 1 * time.Second + stakingTrackerMetrics := metrics.NewBTCStakingTrackerMetrics() + + bsTracker := bst.NewBTCStakingTracker( + tm.BTCClient, + btcNotifier, + tm.BabylonClient, + &bstCfg, + &commonCfg, + zap.NewNop(), + stakingTrackerMetrics, + ) + bsTracker.Start() + defer bsTracker.Stop() + + // set up a finality provider + _, fpSK := tm.CreateFinalityProvider(t) + // set up a BTC delegation + stakingMsgTx, stakingSlashingInfo, unbondingSlashingInfo, delSK := tm.CreateBTCDelegationWithoutIncl(t, fpSK) + stakingMsgTxHash := stakingMsgTx.TxHash() + + // send staking tx to Bitcoin node's mempool + _, err = tm.BTCClient.SendRawTransaction(stakingMsgTx, true) + require.NoError(t, err) + + require.Eventually(t, func() bool { + return len(tm.RetrieveTransactionFromMempool(t, []*chainhash.Hash{&stakingMsgTxHash})) == 1 + }, eventuallyWaitTimeOut, eventuallyPollTime) + + // 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, + ) + require.NoError(t, err) + unbondingSlashingInfo.UnbondingTx.TxIn[0].Witness = witness + + // Send unbonding tx to Bitcoin + _, err = tm.BTCClient.SendRawTransaction(unbondingSlashingInfo.UnbondingTx, true) + require.NoError(t, err) + + 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) + + mBlock := tm.mineBlock(t) + // both staking and unbonding txs are in this block + require.Equal(t, 3, len(mBlock.Transactions)) + + // insert k empty blocks to Bitcoin + btccParamsResp, err := tm.BabylonClient.BTCCheckpointParams() + require.NoError(t, err) + btccParams := btccParamsResp.Params + for i := 0; i < int(btccParams.BtcConfirmationDepth); i++ { + tm.mineBlock(t) + } + + tm.CatchUpBTCLightClient(t) + + // wait until delegation has become unbonded + require.Eventually(t, func() bool { + resp, err := tm.BabylonClient.BTCDelegation(stakingSlashingInfo.StakingTx.TxHash().String()) + require.NoError(t, err) + + return resp.BtcDelegation.StatusDesc == btcstakingtypes.BTCDelegationStatus_UNBONDED.String() + }, eventuallyWaitTimeOut, eventuallyPollTime) +} From 2f5f9b859c3b27979f438133168537525c508262 Mon Sep 17 00:00:00 2001 From: lazar Date: Fri, 4 Oct 2024 14:06:08 +0200 Subject: [PATCH 2/7] build flags --- e2etest/unbondingwatcher_e2e_test.go | 3 +++ 1 file changed, 3 insertions(+) diff --git a/e2etest/unbondingwatcher_e2e_test.go b/e2etest/unbondingwatcher_e2e_test.go index 94d3845..1daedae 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 9294efefca1c73ef26921b0c1d41909c47b4e952 Mon Sep 17 00:00:00 2001 From: lazar Date: Fri, 4 Oct 2024 14:16:23 +0200 Subject: [PATCH 3/7] cleanup --- btcstaking-tracker/stakingeventwatcher/tracked_delegations.go | 1 - 1 file changed, 1 deletion(-) diff --git a/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go b/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go index 63ec7a2..cd9a019 100644 --- a/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go +++ b/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go @@ -113,7 +113,6 @@ func (td *TrackedDelegations) HasDelegationChanged( // Compare fields to check if the delegation has changed if existingDelegation.StakingOutputIdx != newDelegation.stakingOutputIdx || - existingDelegation.StakingTx.TxHash() != newDelegation.stakingTx.TxHash() || !reflect.DeepEqual(existingDelegation.UnbondingOutput, newDelegation.unbondingOutput) || existingDelegation.DelegationStartHeight != newDelegation.delegationStartHeight { return true, true // The delegation has changed and it exists From 885e1191263a106b5414cb1bf58cd83e088b55d7 Mon Sep 17 00:00:00 2001 From: lazar Date: Fri, 4 Oct 2024 14:21:18 +0200 Subject: [PATCH 4/7] housekeeping --- e2etest/unbondingwatcher_e2e_test.go | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/e2etest/unbondingwatcher_e2e_test.go b/e2etest/unbondingwatcher_e2e_test.go index 1daedae..9157bc0 100644 --- a/e2etest/unbondingwatcher_e2e_test.go +++ b/e2etest/unbondingwatcher_e2e_test.go @@ -201,7 +201,9 @@ func TestActivatingDelegation(t *testing.T) { } // TestActivatingAndUnbondingDelegation tests that delegation will eventually become UNBONDED given that -// both staking and unbonding tx are in the same block. In this test +// both staking and unbonding tx are in the same block. +// 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.Parallel() // segwit is activated at height 300. It's necessary for staking/slashing tx From 6ed20c4b6f17c1ce3e58e1b2a9491f9f8a501940 Mon Sep 17 00:00:00 2001 From: lazar Date: Fri, 4 Oct 2024 14:41:26 +0200 Subject: [PATCH 5/7] test --- e2etest/unbondingwatcher_e2e_test.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/e2etest/unbondingwatcher_e2e_test.go b/e2etest/unbondingwatcher_e2e_test.go index 9157bc0..f87490a 100644 --- a/e2etest/unbondingwatcher_e2e_test.go +++ b/e2etest/unbondingwatcher_e2e_test.go @@ -205,7 +205,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.Parallel() + //t.Parallel() // segwit is activated at height 300. It's necessary for staking/slashing tx numMatureOutputs := uint32(300) From 530cb455f93771b73043934ee91a95195f349da7 Mon Sep 17 00:00:00 2001 From: lazar Date: Fri, 4 Oct 2024 19:03:41 +0200 Subject: [PATCH 6/7] pr comment --- btcstaking-tracker/stakingeventwatcher/tracked_delegations.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go b/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go index cd9a019..b30ff49 100644 --- a/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go +++ b/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go @@ -100,7 +100,7 @@ func (td *TrackedDelegations) RemoveDelegation(stakingTxHash chainhash.Hash) { func (td *TrackedDelegations) HasDelegationChanged( stakingTxHash chainhash.Hash, newDelegation *newDelegation, -) (bool, bool) { +) (exists bool, changed bool) { td.mu.Lock() defer td.mu.Unlock() From 0566ea099d139ac9744edf3625d3eb191bc137d6 Mon Sep 17 00:00:00 2001 From: lazar Date: Mon, 7 Oct 2024 10:52:04 +0200 Subject: [PATCH 7/7] pr comment --- .../stakingeventwatcher/tracked_delegations.go | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go b/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go index b30ff49..d11e050 100644 --- a/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go +++ b/btcstaking-tracker/stakingeventwatcher/tracked_delegations.go @@ -2,7 +2,6 @@ package stakingeventwatcher import ( "fmt" - "reflect" "sync" "github.com/btcsuite/btcd/chaincfg/chainhash" @@ -111,10 +110,8 @@ func (td *TrackedDelegations) HasDelegationChanged( return false, false } - // Compare fields to check if the delegation has changed - if existingDelegation.StakingOutputIdx != newDelegation.stakingOutputIdx || - !reflect.DeepEqual(existingDelegation.UnbondingOutput, newDelegation.unbondingOutput) || - existingDelegation.DelegationStartHeight != newDelegation.delegationStartHeight { + // Compare height to check if the delegation has changed + if existingDelegation.DelegationStartHeight != newDelegation.delegationStartHeight { return true, true // The delegation has changed and it exists }