Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

chore(e2e): e2e test for activating and unbonded in mempool #71

Merged
merged 7 commits into from
Oct 7, 2024
Merged
Show file tree
Hide file tree
Changes from 5 commits
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
23 changes: 16 additions & 7 deletions btcstaking-tracker/stakingeventwatcher/stakingeventwatcher.go
Original file line number Diff line number Diff line change
Expand Up @@ -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,
Expand All @@ -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)
Copy link
Member Author

@Lazar955 Lazar955 Oct 4, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the delegation might have changed from verified to active status, in order to track unbonding, we need the activation height, and data in the tracker might be stale, so let's check this

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this was a bit confusing the checks exists && changed or !exists

If the delegation didn't exists, we still push it as a new delegation to the unbondingDelegationChan?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, we want to track it, but we also want to push it to the channel if it exits and has changed (e.g went from verified -> activated)

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,
Expand All @@ -237,6 +242,8 @@ func (sew *StakingEventWatcher) fetchDelegations() {
del.stakingTx,
del.stakingOutputIdx,
del.unbondingOutput,
del.delegationStartHeight,
false,
)
}
}
Expand All @@ -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)
}
}()
Expand Down Expand Up @@ -457,6 +464,8 @@ func (sew *StakingEventWatcher) handleUnbondedDelegations() {
activeDel.stakingTx,
activeDel.stakingOutputIdx,
activeDel.unbondingOutput,
activeDel.delegationStartHeight,
true,
)

if err != nil {
Expand Down
47 changes: 41 additions & 6 deletions btcstaking-tracker/stakingeventwatcher/tracked_delegations.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,16 +2,18 @@ package stakingeventwatcher

import (
"fmt"
"reflect"
"sync"

"github.com/btcsuite/btcd/chaincfg/chainhash"
"github.com/btcsuite/btcd/wire"
)

type TrackedDelegation struct {
StakingTx *wire.MsgTx
StakingOutputIdx uint32
UnbondingOutput *wire.TxOut
StakingTx *wire.MsgTx
StakingOutputIdx uint32
UnbondingOutput *wire.TxOut
DelegationStartHeight uint64
}

type TrackedDelegations struct {
Expand Down Expand Up @@ -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()
Expand All @@ -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)
}

Expand All @@ -86,3 +96,28 @@ func (td *TrackedDelegations) RemoveDelegation(stakingTxHash chainhash.Hash) {

delete(td.mapping, stakingTxHash)
}

func (td *TrackedDelegations) HasDelegationChanged(
stakingTxHash chainhash.Hash,
newDelegation *newDelegation,
) (bool, bool) {
Lazar955 marked this conversation as resolved.
Show resolved Hide resolved
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 ||
!reflect.DeepEqual(existingDelegation.UnbondingOutput, newDelegation.unbondingOutput) ||
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

hmm I think the only thing that could have changes is .DelegationStartHeight ? I.e

  • .StakingOutputIdx will always be the same
  • .UnbondingOutput will always be the same
  • .DelegationStartHeight can change from 0 to some other value when providing proof of inclusion

So maybe lets be explicit here at compare only DelegationStartHeight ?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

sg will amend

existingDelegation.DelegationStartHeight != newDelegation.delegationStartHeight {
return true, true // The delegation has changed and it exists
}

// The delegation exists but hasn't changed
return false, true
}
31 changes: 2 additions & 29 deletions e2etest/test_manager_btcstaking.go
Original file line number Diff line number Diff line change
Expand Up @@ -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)

Expand Down Expand Up @@ -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(
Expand Down
145 changes: 144 additions & 1 deletion e2etest/unbondingwatcher_e2e_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
package e2etest

import (
btcstakingtypes "github.com/babylonlabs-io/babylon/x/btcstaking/types"
"go.uber.org/zap"
"testing"
"time"
Expand Down Expand Up @@ -157,7 +158,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,
Expand All @@ -170,3 +199,117 @@ 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, 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()
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

for some reason, there might be an unwanted dependancy between e2e test, making it sequential until, will investigate (related to btc frequent headers maybe)

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

"works on my machine" type of thing 😅

// 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)
}
Loading