From 823b5d02095b37f668ec67362f466b06d6ef592d Mon Sep 17 00:00:00 2001 From: KonradStaniec Date: Tue, 15 Oct 2024 16:37:49 +0200 Subject: [PATCH 01/11] Refactor stakerApp logic in regards to two flows --- itest/e2e_test.go | 26 +-- staker/events.go | 121 ---------- staker/stakerapp.go | 348 +++++++++++++---------------- staker/types.go | 4 +- stakerdb/trackedtranactionstore.go | 100 ++++++++- 5 files changed, 275 insertions(+), 324 deletions(-) diff --git a/itest/e2e_test.go b/itest/e2e_test.go index 18dd2cf..02e0748 100644 --- a/itest/e2e_test.go +++ b/itest/e2e_test.go @@ -716,7 +716,7 @@ func (tm *TestManager) sendStakingTxBTC( require.Equal(t, stakingDetails.StakingTxHash, txHash) if sendToBabylonFirst { - require.Equal(t, stakingDetails.StakingState, proto.TransactionState_TRANSACTION_CREATED.String()) + require.Equal(t, stakingDetails.StakingState, proto.TransactionState_SENT_TO_BABYLON.String()) } else { require.Equal(t, stakingDetails.StakingState, proto.TransactionState_SENT_TO_BTC.String()) } @@ -1129,7 +1129,7 @@ func (tm *TestManager) insertCovenantSigForDelegation( require.NoError(t, err) } -func TestStakingFailures(t *testing.T) { +func ATestStakingFailures(t *testing.T) { t.Parallel() numMatureOutputs := uint32(200) ctx, cancel := context.WithCancel(context.Background()) @@ -1245,7 +1245,7 @@ func TestSendingStakingTransaction(t *testing.T) { require.Equal(t, transactionsResult.Transactions[0].StakingTxHash, txHash.String()) } -func TestSendingStakingTransactionWithPreApproval(t *testing.T) { +func ATestSendingStakingTransactionWithPreApproval(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1312,7 +1312,7 @@ func TestSendingStakingTransactionWithPreApproval(t *testing.T) { } -func TestMultipleWithdrawableStakingTransactions(t *testing.T) { +func ATestMultipleWithdrawableStakingTransactions(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1380,7 +1380,7 @@ func TestMultipleWithdrawableStakingTransactions(t *testing.T) { require.Equal(t, withdrawableTransactionsResp.Transactions[2].TransactionIdx, "4") } -func TestSendingWatchedStakingTransaction(t *testing.T) { +func ATestSendingWatchedStakingTransaction(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1404,7 +1404,7 @@ func TestSendingWatchedStakingTransaction(t *testing.T) { tm.waitForStakingTxState(t, txHash, proto.TransactionState_SENT_TO_BABYLON) } -func TestRestartingTxNotDeepEnough(t *testing.T) { +func ATestRestartingTxNotDeepEnough(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1433,7 +1433,7 @@ func TestRestartingTxNotDeepEnough(t *testing.T) { tm.waitForStakingTxState(t, txHash, proto.TransactionState_SENT_TO_BABYLON) } -func TestRestartingTxNotOnBabylon(t *testing.T) { +func ATestRestartingTxNotOnBabylon(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1478,7 +1478,7 @@ func TestRestartingTxNotOnBabylon(t *testing.T) { } } -func TestStakingUnbonding(t *testing.T) { +func ATestStakingUnbonding(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1550,7 +1550,7 @@ func TestStakingUnbonding(t *testing.T) { tm.waitForStakingTxState(t, txHash, proto.TransactionState_SPENT_ON_BTC) } -func TestUnbondingRestartWaitingForSignatures(t *testing.T) { +func ATestUnbondingRestartWaitingForSignatures(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1625,7 +1625,7 @@ func containsOutput(outputs []walletcontroller.Utxo, address string, amount btcu return false } -func TestBitcoindWalletRpcApi(t *testing.T) { +func ATestBitcoindWalletRpcApi(t *testing.T) { t.Parallel() manager, err := containers.NewManager(t) require.NoError(t, err) @@ -1701,7 +1701,7 @@ func TestBitcoindWalletRpcApi(t *testing.T) { require.Equal(t, walletcontroller.TxInChain, status) } -func TestBitcoindWalletBip322Signing(t *testing.T) { +func ATestBitcoindWalletBip322Signing(t *testing.T) { t.Parallel() manager, err := containers.NewManager(t) require.NoError(t, err) @@ -1732,7 +1732,7 @@ func TestBitcoindWalletBip322Signing(t *testing.T) { require.NoError(t, err) } -func TestSendingStakingTransaction_Restaking(t *testing.T) { +func ATestSendingStakingTransaction_Restaking(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1774,7 +1774,7 @@ func TestSendingStakingTransaction_Restaking(t *testing.T) { tm.waitForStakingTxState(t, txHash, proto.TransactionState_DELEGATION_ACTIVE) } -func TestRecoverAfterRestartDuringWithdrawal(t *testing.T) { +func ATestRecoverAfterRestartDuringWithdrawal(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs diff --git a/staker/events.go b/staker/events.go index 81a3cbd..b171e52 100644 --- a/staker/events.go +++ b/staker/events.go @@ -2,12 +2,8 @@ package staker import ( cl "github.com/babylonlabs-io/btc-staker/babylonclient" - "github.com/btcsuite/btcd/btcec/v2" - "github.com/btcsuite/btcd/btcec/v2/schnorr" - "github.com/btcsuite/btcd/btcutil" "github.com/btcsuite/btcd/chaincfg/chainhash" "github.com/btcsuite/btcd/wire" - sdk "github.com/cosmos/cosmos-sdk/types" "github.com/sirupsen/logrus" ) @@ -22,7 +18,6 @@ type StakingEvent interface { EventDesc() string } -var _ StakingEvent = (*stakingRequestedEvent)(nil) var _ StakingEvent = (*stakingTxBtcConfirmedEvent)(nil) var _ StakingEvent = (*delegationSubmittedToBabylonEvent)(nil) var _ StakingEvent = (*delegationActiveOnBabylonEvent)(nil) @@ -32,122 +27,6 @@ var _ StakingEvent = (*spendStakeTxConfirmedOnBtcEvent)(nil) var _ StakingEvent = (*sendStakingTxToBTCRequestedEvent)(nil) var _ StakingEvent = (*criticalErrorEvent)(nil) -type stakingRequestedEvent struct { - stakerAddress btcutil.Address - stakingTxHash chainhash.Hash - stakingTx *wire.MsgTx - stakingOutputIdx uint32 - stakingOutputPkScript []byte - stakingTime uint16 - stakingValue btcutil.Amount - fpBtcPks []*btcec.PublicKey - requiredDepthOnBtcChain uint32 - pop *cl.BabylonPop - watchTxData *watchTxData - usePreApprovalFlow bool - errChan chan error - successChan chan *chainhash.Hash -} - -func (req *stakingRequestedEvent) isWatched() bool { - return req.watchTxData != nil -} - -func newOwnedStakingRequest( - stakerAddress btcutil.Address, - stakingTx *wire.MsgTx, - stakingOutputIdx uint32, - stakingOutputPkScript []byte, - stakingTime uint16, - stakingValue btcutil.Amount, - fpBtcPks []*btcec.PublicKey, - confirmationTimeBlocks uint32, - pop *cl.BabylonPop, - usePreApprovalFlow bool, -) *stakingRequestedEvent { - return &stakingRequestedEvent{ - stakerAddress: stakerAddress, - stakingTxHash: stakingTx.TxHash(), - stakingTx: stakingTx, - stakingOutputIdx: stakingOutputIdx, - stakingOutputPkScript: stakingOutputPkScript, - stakingTime: stakingTime, - stakingValue: stakingValue, - fpBtcPks: fpBtcPks, - requiredDepthOnBtcChain: confirmationTimeBlocks, - pop: pop, - watchTxData: nil, - usePreApprovalFlow: usePreApprovalFlow, - errChan: make(chan error, 1), - successChan: make(chan *chainhash.Hash, 1), - } -} - -type watchTxData struct { - slashingTx *wire.MsgTx - slashingTxSig *schnorr.Signature - stakerBabylonAddr sdk.AccAddress - stakerBtcPk *btcec.PublicKey - // unbonding related data - unbondingTx *wire.MsgTx - slashUnbondingTx *wire.MsgTx - slashUnbondingTxSig *schnorr.Signature - unbondingTime uint16 -} - -func newWatchedStakingRequest( - stakerAddress btcutil.Address, - stakingTx *wire.MsgTx, - stakingOutputIdx uint32, - stakingOutputPkScript []byte, - stakingTime uint16, - stakingValue btcutil.Amount, - fpBtcPks []*btcec.PublicKey, - confirmationTimeBlocks uint32, - pop *cl.BabylonPop, - slashingTx *wire.MsgTx, - slashingTxSignature *schnorr.Signature, - stakerBabylonAddr sdk.AccAddress, - stakerBtcPk *btcec.PublicKey, - unbondingTx *wire.MsgTx, - slashUnbondingTx *wire.MsgTx, - slashUnbondingTxSig *schnorr.Signature, - unbondingTime uint16, -) *stakingRequestedEvent { - return &stakingRequestedEvent{ - stakerAddress: stakerAddress, - stakingTxHash: stakingTx.TxHash(), - stakingTx: stakingTx, - stakingOutputIdx: stakingOutputIdx, - stakingOutputPkScript: stakingOutputPkScript, - stakingTime: stakingTime, - stakingValue: stakingValue, - fpBtcPks: fpBtcPks, - requiredDepthOnBtcChain: confirmationTimeBlocks, - pop: pop, - watchTxData: &watchTxData{ - slashingTx: slashingTx, - slashingTxSig: slashingTxSignature, - stakerBabylonAddr: stakerBabylonAddr, - stakerBtcPk: stakerBtcPk, - unbondingTx: unbondingTx, - slashUnbondingTx: slashUnbondingTx, - slashUnbondingTxSig: slashUnbondingTxSig, - unbondingTime: unbondingTime, - }, - errChan: make(chan error, 1), - successChan: make(chan *chainhash.Hash, 1), - } -} - -func (req *stakingRequestedEvent) EventId() chainhash.Hash { - return req.stakingTxHash -} - -func (req *stakingRequestedEvent) EventDesc() string { - return "STAKING_REQUESTED" -} - type stakingTxBtcConfirmedEvent struct { stakingTxHash chainhash.Hash txIndex uint32 diff --git a/staker/stakerapp.go b/staker/stakerapp.go index 99a6d30..3813bb1 100644 --- a/staker/stakerapp.go +++ b/staker/stakerapp.go @@ -123,8 +123,7 @@ type StakerApp struct { babylonMsgSender *cl.BabylonMsgSender m *metrics.StakerMetrics - stakingRequestedEvChan chan *stakingRequestedEvent - sendStakingTxToBTCRequestedEvChan chan *sendStakingTxToBTCRequestedEvent + stakingRequestedCmdChan chan *stakingRequestCmd stakingTxBtcConfirmedEvChan chan *stakingTxBtcConfirmedEvent delegationSubmittedToBabylonEvChan chan *delegationSubmittedToBabylonEvent delegationActiveOnBabylonEvChan chan *delegationActiveOnBabylonEvent @@ -218,21 +217,18 @@ func NewStakerAppFromDeps( metrics *metrics.StakerMetrics, ) (*StakerApp, error) { return &StakerApp{ - babylonClient: cl, - wc: walletClient, - notifier: nodeNotifier, - feeEstimator: feeEestimator, - network: &config.ActiveNetParams, - txTracker: tracker, - babylonMsgSender: babylonMsgSender, - m: metrics, - config: config, - logger: logger, - quit: make(chan struct{}), - stakingRequestedEvChan: make(chan *stakingRequestedEvent), - - sendStakingTxToBTCRequestedEvChan: make(chan *sendStakingTxToBTCRequestedEvent), - + babylonClient: cl, + wc: walletClient, + notifier: nodeNotifier, + feeEstimator: feeEestimator, + network: &config.ActiveNetParams, + txTracker: tracker, + babylonMsgSender: babylonMsgSender, + m: metrics, + config: config, + logger: logger, + quit: make(chan struct{}), + stakingRequestedCmdChan: make(chan *stakingRequestCmd), // event for when transaction is confirmed on BTC stakingTxBtcConfirmedEvChan: make(chan *stakingTxBtcConfirmedEvent), @@ -306,9 +302,10 @@ func (app *StakerApp) Start() error { app.babylonMsgSender.Start() - app.wg.Add(2) + app.wg.Add(3) go app.handleNewBlocks(blockEventNotifier) go app.handleStakingEvents() + go app.handleStakingCommands() if err := app.checkTransactionsStatus(); err != nil { startErr = err @@ -1339,202 +1336,178 @@ func (app *StakerApp) sendDelegationToBabylonTask( } } -// main event loop for the staker app -func (app *StakerApp) handleStakingEvents() { +func (app *StakerApp) handlePreApprovalCmd(cmd *stakingRequestCmd) error { + // just to pass to buildAndSendDelegation + fakeStoredTx, err := stakerdb.CreateTrackedTransaction( + cmd.stakingTx, + cmd.stakingOutputIdx, + cmd.stakingTime, + cmd.fpBtcPks, + babylonPopToDbPop(cmd.pop), + cmd.stakerAddress, + ) + + if err != nil { + return err + } + + req := &sendDelegationRequest{ + txHash: cmd.stakingTxHash, + inclusionInfo: nil, + requiredInclusionBlockDepth: cmd.requiredDepthOnBtcChain, + } + + _, delegationData, err := app.buildAndSendDelegation( + req, + cmd.stakerAddress, + fakeStoredTx, + ) + + if err != nil { + return err + } + + err = app.txTracker.AddTransactionSentToBabylon( + cmd.stakingTx, + cmd.stakingOutputIdx, + cmd.stakingTime, + cmd.fpBtcPks, + babylonPopToDbPop(cmd.pop), + cmd.stakerAddress, + delegationData.Ud.UnbondingTransaction, + delegationData.Ud.UnbondingTxUnbondingTime, + ) + + if err != nil { + return err + } + + app.wg.Add(1) + go app.checkForUnbondingTxSignaturesOnBabylon(&cmd.stakingTxHash) + + return nil +} + +func (app *StakerApp) handlePostApprovalCmd(cmd *stakingRequestCmd) error { + bestBlockHeight := app.currentBestBlockHeight.Load() + + _, err := app.wc.SendRawTransaction(cmd.stakingTx, true) + + if err != nil { + return err + } + + stakingOutputPkScript := cmd.stakingTx.TxOut[cmd.stakingOutputIdx].PkScript + + if err := app.waitForStakingTransactionConfirmation( + &cmd.stakingTxHash, + stakingOutputPkScript, + cmd.requiredDepthOnBtcChain, + uint32(bestBlockHeight), + ); err != nil { + return err + } + + if err := app.txTracker.AddTransaction( + cmd.stakingTx, + cmd.stakingOutputIdx, + cmd.stakingTime, + cmd.fpBtcPks, + babylonPopToDbPop(cmd.pop), + cmd.stakerAddress, + ); err != nil { + return err + } + + return nil +} + +func (app *StakerApp) handleStakingCmd(cmd *stakingRequestCmd) error { + if cmd.usePreApprovalFlow { + return app.handlePreApprovalCmd(cmd) + } else { + return app.handlePostApprovalCmd(cmd) + } +} + +func (app *StakerApp) handleStakingCommands() { defer app.wg.Done() for { select { - case ev := <-app.stakingRequestedEvChan: - app.logStakingEventReceived(ev) + case cmd := <-app.stakingRequestedCmdChan: + app.logStakingEventReceived(cmd) - if ev.isWatched() { + if cmd.isWatched() { bestBlockHeight := app.currentBestBlockHeight.Load() err := app.txTracker.AddWatchedTransaction( - ev.stakingTx, - ev.stakingOutputIdx, - ev.stakingTime, - ev.fpBtcPks, - babylonPopToDbPop(ev.pop), - ev.stakerAddress, - ev.watchTxData.slashingTx, - ev.watchTxData.slashingTxSig, - ev.watchTxData.stakerBabylonAddr, - ev.watchTxData.stakerBtcPk, - ev.watchTxData.unbondingTx, - ev.watchTxData.slashUnbondingTx, - ev.watchTxData.slashUnbondingTxSig, - ev.watchTxData.unbondingTime, + cmd.stakingTx, + cmd.stakingOutputIdx, + cmd.stakingTime, + cmd.fpBtcPks, + babylonPopToDbPop(cmd.pop), + cmd.stakerAddress, + cmd.watchTxData.slashingTx, + cmd.watchTxData.slashingTxSig, + cmd.watchTxData.stakerBabylonAddr, + cmd.watchTxData.stakerBtcPk, + cmd.watchTxData.unbondingTx, + cmd.watchTxData.slashUnbondingTx, + cmd.watchTxData.slashUnbondingTxSig, + cmd.watchTxData.unbondingTime, ) if err != nil { - ev.errChan <- err + cmd.errChan <- err continue } // we assume tx is already on btc chain, so we need to wait for confirmation if err := app.waitForStakingTransactionConfirmation( - &ev.stakingTxHash, - ev.stakingTx.TxOut[ev.stakingOutputIdx].PkScript, - ev.requiredDepthOnBtcChain, + &cmd.stakingTxHash, + cmd.stakingTx.TxOut[cmd.stakingOutputIdx].PkScript, + cmd.requiredDepthOnBtcChain, uint32(bestBlockHeight), ); err != nil { - ev.errChan <- err + cmd.errChan <- err continue } app.m.ValidReceivedDelegationRequests.Inc() - ev.successChan <- &ev.stakingTxHash - } else { - err := app.txTracker.AddTransaction( - ev.stakingTx, - ev.stakingOutputIdx, - ev.stakingTime, - ev.fpBtcPks, - babylonPopToDbPop(ev.pop), - ev.stakerAddress, - ) - - if err != nil { - ev.errChan <- err - continue - } - - app.logger.Info("Recieved staking event", "ususePreApprovalFlowe", ev.usePreApprovalFlow) - - if ev.usePreApprovalFlow { - req := &sendDelegationRequest{ - txHash: ev.stakingTxHash, - inclusionInfo: nil, - requiredInclusionBlockDepth: ev.requiredDepthOnBtcChain, - } - - storedTx, stakerAddress := app.mustGetTransactionAndStakerAddress(&ev.stakingTxHash) - - app.wg.Add(1) - go func( - req *sendDelegationRequest, - stakerAddress btcutil.Address, - storedTx *stakerdb.StoredTransaction, - ev *stakingRequestedEvent, - ) { - defer app.wg.Done() - _, delegationData, err := app.buildAndSendDelegation( - req, - stakerAddress, - storedTx, - ) - - if err != nil { - utils.PushOrQuit( - ev.errChan, - err, - app.quit, - ) - return - } - - submittedEv := &delegationSubmittedToBabylonEvent{ - stakingTxHash: req.txHash, - unbondingTx: delegationData.Ud.UnbondingTransaction, - unbondingTime: delegationData.Ud.UnbondingTxUnbondingTime, - } - - // push event to channel to start waiting for covenant signatures - utils.PushOrQuit[*delegationSubmittedToBabylonEvent]( - app.delegationSubmittedToBabylonEvChan, - submittedEv, - app.quit, - ) - - // send success to caller - utils.PushOrQuit( - ev.successChan, - &ev.stakingTxHash, - app.quit, - ) - }(req, stakerAddress, storedTx, ev) - } else { - // old flow, send to BTC first, end expect response to the caller - app.wg.Add(1) - go func() { - defer app.wg.Done() - utils.PushOrQuit( - app.sendStakingTxToBTCRequestedEvChan, - &sendStakingTxToBTCRequestedEvent{ - stakingTxHash: ev.stakingTxHash, - requiredDepthOnBtcChain: ev.requiredDepthOnBtcChain, - responseExpected: &responseExpectedChan{ - errChan: ev.errChan, - successChan: ev.successChan, - }, - }, - app.quit, - ) - }() - } - app.logStakingEventProcessed(ev) - } - - case ev := <-app.sendStakingTxToBTCRequestedEvChan: - app.logStakingEventReceived(ev) - - bestBlockHeight := app.currentBestBlockHeight.Load() - - storedTx, _ := app.mustGetTransactionAndStakerAddress(&ev.stakingTxHash) - - _, err := app.wc.SendRawTransaction(storedTx.StakingTx, true) - - if err != nil { - if ev.responseExpected != nil { - utils.PushOrQuit( - ev.responseExpected.errChan, - err, - app.quit, - ) - } - app.logStakingEventProcessed(ev) + cmd.successChan <- &cmd.stakingTxHash + app.logStakingEventProcessed(cmd) continue } - if err := app.txTracker.SetTxSentToBtc( - &ev.stakingTxHash, - ); err != nil { - // TODO: handle this error somehow, it means we received confirmation for tx which we do not store - // which is seems like programming error. Maybe panic? - app.logger.Fatalf("Error setting state for tx %s: %s", ev.stakingTxHash, err) - } - - stakingOutputPkScript := storedTx.StakingTx.TxOut[storedTx.StakingOutputIndex].PkScript + err := app.handleStakingCmd(cmd) - if err := app.waitForStakingTransactionConfirmation( - &ev.stakingTxHash, - stakingOutputPkScript, - ev.requiredDepthOnBtcChain, - uint32(bestBlockHeight), - ); err != nil { - if ev.responseExpected != nil { - utils.PushOrQuit( - ev.responseExpected.errChan, - err, - app.quit, - ) - } - app.logStakingEventProcessed(ev) - continue - } - - if ev.responseExpected != nil { + if err != nil { utils.PushOrQuit( - ev.responseExpected.successChan, - &ev.stakingTxHash, + cmd.errChan, + err, + app.quit, + ) + } else { + utils.PushOrQuit( + cmd.successChan, + &cmd.stakingTxHash, app.quit, ) } + app.logStakingEventProcessed(cmd) + case <-app.quit: + return + } + } +} - app.logStakingEventProcessed(ev) +// main event loop for the staker app +func (app *StakerApp) handleStakingEvents() { + defer app.wg.Done() + for { + select { case ev := <-app.stakingTxBtcConfirmedEvChan: app.logStakingEventReceived(ev) @@ -1759,8 +1732,8 @@ func (app *StakerApp) WatchStaking( "btxTxHash": stakingTx.TxHash(), }).Info("Received valid staking tx to watch") - utils.PushOrQuit[*stakingRequestedEvent]( - app.stakingRequestedEvChan, + utils.PushOrQuit[*stakingRequestCmd]( + app.stakingRequestedCmdChan, watchedRequest, app.quit, ) @@ -1882,6 +1855,7 @@ func (app *StakerApp) StakeFunds( feeRate := app.feeEstimator.EstimateFeePerKb() + // Create unsigned transaction by wallet tx, err := app.wc.CreateAndSignTx([]*wire.TxOut{stakingInfo.StakingOutput}, btcutil.Amount(feeRate), stakerAddress) if err != nil { @@ -1895,7 +1869,7 @@ func (app *StakerApp) StakeFunds( "fee": feeRate, }).Info("Created and signed staking transaction") - req := newOwnedStakingRequest( + req := newOwnedStakingCommand( stakerAddress, tx, 0, @@ -1908,8 +1882,8 @@ func (app *StakerApp) StakeFunds( sendToBabylonFirst, ) - utils.PushOrQuit[*stakingRequestedEvent]( - app.stakingRequestedEvChan, + utils.PushOrQuit[*stakingRequestCmd]( + app.stakingRequestedCmdChan, req, app.quit, ) diff --git a/staker/types.go b/staker/types.go index 4a2e803..6a59991 100644 --- a/staker/types.go +++ b/staker/types.go @@ -479,7 +479,7 @@ func parseWatchStakingRequest( unbondingTime uint16, currentParams *cl.StakingParams, network *chaincfg.Params, -) (*stakingRequestedEvent, error) { +) (*stakingRequestCmd, error) { // TODO(https://github.com/babylonlabs-io/btc-staker/issues/32): // This check re-implements whole babylon validation logic. We should // refactor this to use babylon validation utilities. @@ -634,7 +634,7 @@ func parseWatchStakingRequest( return nil, fmt.Errorf("failed to watch staking tx. Unbonding tx do not point to staking tx") } - req := newWatchedStakingRequest( + req := newWatchedStakingCmd( stakerAddress, stakingTx, uint32(stakingOutputIdx), diff --git a/stakerdb/trackedtranactionstore.go b/stakerdb/trackedtranactionstore.go index 8ca56a1..bad1943 100644 --- a/stakerdb/trackedtranactionstore.go +++ b/stakerdb/trackedtranactionstore.go @@ -552,6 +552,104 @@ func (c *TrackedTransactionStore) addTransactionInternal( }) } +func CreateTrackedTransaction( + btcTx *wire.MsgTx, + stakingOutputIndex uint32, + stakingTime uint16, + fpPubKeys []*btcec.PublicKey, + pop *ProofOfPossession, + stakerAddress btcutil.Address, +) (*StoredTransaction, error) { + serializedTx, err := utils.SerializeBtcTransaction(btcTx) + + if err != nil { + return nil, err + } + + if len(fpPubKeys) == 0 { + return nil, fmt.Errorf("cannot add transaction without finality providers public keys") + } + + var fpPubKeysBytes [][]byte = make([][]byte, len(fpPubKeys)) + + for i, pk := range fpPubKeys { + fpPubKeysBytes[i] = schnorr.SerializePubKey(pk) + } + + msg := proto.TrackedTransaction{ + // Setting it to 0, proper number will be filled by `addTransactionInternal` + TrackedTransactionIdx: 0, + StakingTransaction: serializedTx, + StakingOutputIdx: stakingOutputIndex, + StakerAddress: stakerAddress.EncodeAddress(), + StakingTime: uint32(stakingTime), + FinalityProvidersBtcPks: fpPubKeysBytes, + StakingTxBtcConfirmationInfo: nil, + BtcSigType: pop.BtcSigType, + BtcSigOverBbnStakerAddr: pop.BtcSigOverBabylonAddr, + State: proto.TransactionState_TRANSACTION_CREATED, + Watched: false, + UnbondingTxData: nil, + } + + return protoTxToStoredTransaction(&msg) +} + +func (c *TrackedTransactionStore) AddTransactionSentToBabylon( + btcTx *wire.MsgTx, + stakingOutputIndex uint32, + stakingTime uint16, + fpPubKeys []*btcec.PublicKey, + pop *ProofOfPossession, + stakerAddress btcutil.Address, + unbondingTx *wire.MsgTx, + unbondingTime uint16, +) error { + txHash := btcTx.TxHash() + txHashBytes := txHash[:] + serializedTx, err := utils.SerializeBtcTransaction(btcTx) + + if err != nil { + return err + } + + if len(fpPubKeys) == 0 { + return fmt.Errorf("cannot add transaction without finality providers public keys") + } + + var fpPubKeysBytes [][]byte = make([][]byte, len(fpPubKeys)) + + for i, pk := range fpPubKeys { + fpPubKeysBytes[i] = schnorr.SerializePubKey(pk) + } + + update, err := newInitialUnbondingTxData(unbondingTx, unbondingTime) + + if err != nil { + return err + } + + msg := proto.TrackedTransaction{ + // Setting it to 0, proper number will be filled by `addTransactionInternal` + TrackedTransactionIdx: 0, + StakingTransaction: serializedTx, + StakingOutputIdx: stakingOutputIndex, + StakerAddress: stakerAddress.EncodeAddress(), + StakingTime: uint32(stakingTime), + FinalityProvidersBtcPks: fpPubKeysBytes, + StakingTxBtcConfirmationInfo: nil, + BtcSigType: pop.BtcSigType, + BtcSigOverBbnStakerAddr: pop.BtcSigOverBabylonAddr, + State: proto.TransactionState_SENT_TO_BABYLON, + Watched: false, + UnbondingTxData: update, + } + + return c.addTransactionInternal( + txHashBytes, &msg, nil, + ) +} + func (c *TrackedTransactionStore) AddTransaction( btcTx *wire.MsgTx, stakingOutputIndex uint32, @@ -589,7 +687,7 @@ func (c *TrackedTransactionStore) AddTransaction( StakingTxBtcConfirmationInfo: nil, BtcSigType: pop.BtcSigType, BtcSigOverBbnStakerAddr: pop.BtcSigOverBabylonAddr, - State: proto.TransactionState_TRANSACTION_CREATED, + State: proto.TransactionState_SENT_TO_BTC, Watched: false, UnbondingTxData: nil, } From 136285882c36919703557f4a17fcc838014bc695 Mon Sep 17 00:00:00 2001 From: KonradStaniec Date: Tue, 15 Oct 2024 17:07:00 +0200 Subject: [PATCH 02/11] Remove unnecesary state --- itest/e2e_test.go | 24 +++++----- proto/transaction.pb.go | 77 ++++++++++++++---------------- proto/transaction.proto | 15 +++--- staker/stakerapp.go | 77 +----------------------------- stakerdb/trackedtranactionstore.go | 2 +- 5 files changed, 57 insertions(+), 138 deletions(-) diff --git a/itest/e2e_test.go b/itest/e2e_test.go index 02e0748..3b9fc3e 100644 --- a/itest/e2e_test.go +++ b/itest/e2e_test.go @@ -1129,7 +1129,7 @@ func (tm *TestManager) insertCovenantSigForDelegation( require.NoError(t, err) } -func ATestStakingFailures(t *testing.T) { +func TestStakingFailures(t *testing.T) { t.Parallel() numMatureOutputs := uint32(200) ctx, cancel := context.WithCancel(context.Background()) @@ -1245,7 +1245,7 @@ func TestSendingStakingTransaction(t *testing.T) { require.Equal(t, transactionsResult.Transactions[0].StakingTxHash, txHash.String()) } -func ATestSendingStakingTransactionWithPreApproval(t *testing.T) { +func TestSendingStakingTransactionWithPreApproval(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1312,7 +1312,7 @@ func ATestSendingStakingTransactionWithPreApproval(t *testing.T) { } -func ATestMultipleWithdrawableStakingTransactions(t *testing.T) { +func TestMultipleWithdrawableStakingTransactions(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1380,7 +1380,7 @@ func ATestMultipleWithdrawableStakingTransactions(t *testing.T) { require.Equal(t, withdrawableTransactionsResp.Transactions[2].TransactionIdx, "4") } -func ATestSendingWatchedStakingTransaction(t *testing.T) { +func TestSendingWatchedStakingTransaction(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1404,7 +1404,7 @@ func ATestSendingWatchedStakingTransaction(t *testing.T) { tm.waitForStakingTxState(t, txHash, proto.TransactionState_SENT_TO_BABYLON) } -func ATestRestartingTxNotDeepEnough(t *testing.T) { +func TestRestartingTxNotDeepEnough(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1433,7 +1433,7 @@ func ATestRestartingTxNotDeepEnough(t *testing.T) { tm.waitForStakingTxState(t, txHash, proto.TransactionState_SENT_TO_BABYLON) } -func ATestRestartingTxNotOnBabylon(t *testing.T) { +func TestRestartingTxNotOnBabylon(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1478,7 +1478,7 @@ func ATestRestartingTxNotOnBabylon(t *testing.T) { } } -func ATestStakingUnbonding(t *testing.T) { +func TestStakingUnbonding(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1550,7 +1550,7 @@ func ATestStakingUnbonding(t *testing.T) { tm.waitForStakingTxState(t, txHash, proto.TransactionState_SPENT_ON_BTC) } -func ATestUnbondingRestartWaitingForSignatures(t *testing.T) { +func TestUnbondingRestartWaitingForSignatures(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1625,7 +1625,7 @@ func containsOutput(outputs []walletcontroller.Utxo, address string, amount btcu return false } -func ATestBitcoindWalletRpcApi(t *testing.T) { +func TestBitcoindWalletRpcApi(t *testing.T) { t.Parallel() manager, err := containers.NewManager(t) require.NoError(t, err) @@ -1701,7 +1701,7 @@ func ATestBitcoindWalletRpcApi(t *testing.T) { require.Equal(t, walletcontroller.TxInChain, status) } -func ATestBitcoindWalletBip322Signing(t *testing.T) { +func TestBitcoindWalletBip322Signing(t *testing.T) { t.Parallel() manager, err := containers.NewManager(t) require.NoError(t, err) @@ -1732,7 +1732,7 @@ func ATestBitcoindWalletBip322Signing(t *testing.T) { require.NoError(t, err) } -func ATestSendingStakingTransaction_Restaking(t *testing.T) { +func TestSendingStakingTransaction_Restaking(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs @@ -1774,7 +1774,7 @@ func ATestSendingStakingTransaction_Restaking(t *testing.T) { tm.waitForStakingTxState(t, txHash, proto.TransactionState_DELEGATION_ACTIVE) } -func ATestRecoverAfterRestartDuringWithdrawal(t *testing.T) { +func TestRecoverAfterRestartDuringWithdrawal(t *testing.T) { t.Parallel() // need to have at least 300 block on testnet as only then segwit is activated. // Mature output is out which has 100 confirmations, which means 200mature outputs diff --git a/proto/transaction.pb.go b/proto/transaction.pb.go index 97b9eff..9870fd3 100644 --- a/proto/transaction.pb.go +++ b/proto/transaction.pb.go @@ -23,37 +23,34 @@ const ( type TransactionState int32 const ( - TransactionState_TRANSACTION_CREATED TransactionState = 0 - TransactionState_SENT_TO_BTC TransactionState = 1 - TransactionState_CONFIRMED_ON_BTC TransactionState = 2 - TransactionState_SENT_TO_BABYLON TransactionState = 3 - TransactionState_VERIFIED TransactionState = 4 - TransactionState_DELEGATION_ACTIVE TransactionState = 5 - TransactionState_UNBONDING_CONFIRMED_ON_BTC TransactionState = 6 - TransactionState_SPENT_ON_BTC TransactionState = 7 + TransactionState_SENT_TO_BTC TransactionState = 0 + TransactionState_CONFIRMED_ON_BTC TransactionState = 1 + TransactionState_SENT_TO_BABYLON TransactionState = 2 + TransactionState_VERIFIED TransactionState = 3 + TransactionState_DELEGATION_ACTIVE TransactionState = 4 + TransactionState_UNBONDING_CONFIRMED_ON_BTC TransactionState = 5 + TransactionState_SPENT_ON_BTC TransactionState = 6 ) // Enum value maps for TransactionState. var ( TransactionState_name = map[int32]string{ - 0: "TRANSACTION_CREATED", - 1: "SENT_TO_BTC", - 2: "CONFIRMED_ON_BTC", - 3: "SENT_TO_BABYLON", - 4: "VERIFIED", - 5: "DELEGATION_ACTIVE", - 6: "UNBONDING_CONFIRMED_ON_BTC", - 7: "SPENT_ON_BTC", + 0: "SENT_TO_BTC", + 1: "CONFIRMED_ON_BTC", + 2: "SENT_TO_BABYLON", + 3: "VERIFIED", + 4: "DELEGATION_ACTIVE", + 5: "UNBONDING_CONFIRMED_ON_BTC", + 6: "SPENT_ON_BTC", } TransactionState_value = map[string]int32{ - "TRANSACTION_CREATED": 0, - "SENT_TO_BTC": 1, - "CONFIRMED_ON_BTC": 2, - "SENT_TO_BABYLON": 3, - "VERIFIED": 4, - "DELEGATION_ACTIVE": 5, - "UNBONDING_CONFIRMED_ON_BTC": 6, - "SPENT_ON_BTC": 7, + "SENT_TO_BTC": 0, + "CONFIRMED_ON_BTC": 1, + "SENT_TO_BABYLON": 2, + "VERIFIED": 3, + "DELEGATION_ACTIVE": 4, + "UNBONDING_CONFIRMED_ON_BTC": 5, + "SPENT_ON_BTC": 6, } ) @@ -498,7 +495,7 @@ func (x *TrackedTransaction) GetState() TransactionState { if x != nil { return x.State } - return TransactionState_TRANSACTION_CREATED + return TransactionState_SENT_TO_BTC } func (x *TrackedTransaction) GetWatched() bool { @@ -617,23 +614,21 @@ var file_transaction_proto_rawDesc = []byte{ 0x69, 0x6e, 0x67, 0x5f, 0x74, 0x78, 0x5f, 0x64, 0x61, 0x74, 0x61, 0x18, 0x0c, 0x20, 0x01, 0x28, 0x0b, 0x32, 0x16, 0x2e, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x2e, 0x55, 0x6e, 0x62, 0x6f, 0x6e, 0x64, 0x69, 0x6e, 0x67, 0x54, 0x78, 0x44, 0x61, 0x74, 0x61, 0x52, 0x0f, 0x75, 0x6e, 0x62, 0x6f, 0x6e, - 0x64, 0x69, 0x6e, 0x67, 0x54, 0x78, 0x44, 0x61, 0x74, 0x61, 0x2a, 0xbe, 0x01, 0x0a, 0x10, 0x54, + 0x64, 0x69, 0x6e, 0x67, 0x54, 0x78, 0x44, 0x61, 0x74, 0x61, 0x2a, 0xa5, 0x01, 0x0a, 0x10, 0x54, 0x72, 0x61, 0x6e, 0x73, 0x61, 0x63, 0x74, 0x69, 0x6f, 0x6e, 0x53, 0x74, 0x61, 0x74, 0x65, 0x12, - 0x17, 0x0a, 0x13, 0x54, 0x52, 0x41, 0x4e, 0x53, 0x41, 0x43, 0x54, 0x49, 0x4f, 0x4e, 0x5f, 0x43, - 0x52, 0x45, 0x41, 0x54, 0x45, 0x44, 0x10, 0x00, 0x12, 0x0f, 0x0a, 0x0b, 0x53, 0x45, 0x4e, 0x54, - 0x5f, 0x54, 0x4f, 0x5f, 0x42, 0x54, 0x43, 0x10, 0x01, 0x12, 0x14, 0x0a, 0x10, 0x43, 0x4f, 0x4e, - 0x46, 0x49, 0x52, 0x4d, 0x45, 0x44, 0x5f, 0x4f, 0x4e, 0x5f, 0x42, 0x54, 0x43, 0x10, 0x02, 0x12, - 0x13, 0x0a, 0x0f, 0x53, 0x45, 0x4e, 0x54, 0x5f, 0x54, 0x4f, 0x5f, 0x42, 0x41, 0x42, 0x59, 0x4c, - 0x4f, 0x4e, 0x10, 0x03, 0x12, 0x0c, 0x0a, 0x08, 0x56, 0x45, 0x52, 0x49, 0x46, 0x49, 0x45, 0x44, - 0x10, 0x04, 0x12, 0x15, 0x0a, 0x11, 0x44, 0x45, 0x4c, 0x45, 0x47, 0x41, 0x54, 0x49, 0x4f, 0x4e, - 0x5f, 0x41, 0x43, 0x54, 0x49, 0x56, 0x45, 0x10, 0x05, 0x12, 0x1e, 0x0a, 0x1a, 0x55, 0x4e, 0x42, - 0x4f, 0x4e, 0x44, 0x49, 0x4e, 0x47, 0x5f, 0x43, 0x4f, 0x4e, 0x46, 0x49, 0x52, 0x4d, 0x45, 0x44, - 0x5f, 0x4f, 0x4e, 0x5f, 0x42, 0x54, 0x43, 0x10, 0x06, 0x12, 0x10, 0x0a, 0x0c, 0x53, 0x50, 0x45, - 0x4e, 0x54, 0x5f, 0x4f, 0x4e, 0x5f, 0x42, 0x54, 0x43, 0x10, 0x07, 0x42, 0x2c, 0x5a, 0x2a, 0x67, - 0x69, 0x74, 0x68, 0x75, 0x62, 0x2e, 0x63, 0x6f, 0x6d, 0x2f, 0x62, 0x61, 0x62, 0x79, 0x6c, 0x6f, - 0x6e, 0x6c, 0x61, 0x62, 0x73, 0x2d, 0x69, 0x6f, 0x2f, 0x62, 0x74, 0x63, 0x2d, 0x73, 0x74, 0x61, - 0x6b, 0x65, 0x72, 0x2f, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, - 0x33, + 0x0f, 0x0a, 0x0b, 0x53, 0x45, 0x4e, 0x54, 0x5f, 0x54, 0x4f, 0x5f, 0x42, 0x54, 0x43, 0x10, 0x00, + 0x12, 0x14, 0x0a, 0x10, 0x43, 0x4f, 0x4e, 0x46, 0x49, 0x52, 0x4d, 0x45, 0x44, 0x5f, 0x4f, 0x4e, + 0x5f, 0x42, 0x54, 0x43, 0x10, 0x01, 0x12, 0x13, 0x0a, 0x0f, 0x53, 0x45, 0x4e, 0x54, 0x5f, 0x54, + 0x4f, 0x5f, 0x42, 0x41, 0x42, 0x59, 0x4c, 0x4f, 0x4e, 0x10, 0x02, 0x12, 0x0c, 0x0a, 0x08, 0x56, + 0x45, 0x52, 0x49, 0x46, 0x49, 0x45, 0x44, 0x10, 0x03, 0x12, 0x15, 0x0a, 0x11, 0x44, 0x45, 0x4c, + 0x45, 0x47, 0x41, 0x54, 0x49, 0x4f, 0x4e, 0x5f, 0x41, 0x43, 0x54, 0x49, 0x56, 0x45, 0x10, 0x04, + 0x12, 0x1e, 0x0a, 0x1a, 0x55, 0x4e, 0x42, 0x4f, 0x4e, 0x44, 0x49, 0x4e, 0x47, 0x5f, 0x43, 0x4f, + 0x4e, 0x46, 0x49, 0x52, 0x4d, 0x45, 0x44, 0x5f, 0x4f, 0x4e, 0x5f, 0x42, 0x54, 0x43, 0x10, 0x05, + 0x12, 0x10, 0x0a, 0x0c, 0x53, 0x50, 0x45, 0x4e, 0x54, 0x5f, 0x4f, 0x4e, 0x5f, 0x42, 0x54, 0x43, + 0x10, 0x06, 0x42, 0x2c, 0x5a, 0x2a, 0x67, 0x69, 0x74, 0x68, 0x75, 0x62, 0x2e, 0x63, 0x6f, 0x6d, + 0x2f, 0x62, 0x61, 0x62, 0x79, 0x6c, 0x6f, 0x6e, 0x6c, 0x61, 0x62, 0x73, 0x2d, 0x69, 0x6f, 0x2f, + 0x62, 0x74, 0x63, 0x2d, 0x73, 0x74, 0x61, 0x6b, 0x65, 0x72, 0x2f, 0x70, 0x72, 0x6f, 0x74, 0x6f, + 0x62, 0x06, 0x70, 0x72, 0x6f, 0x74, 0x6f, 0x33, } var ( diff --git a/proto/transaction.proto b/proto/transaction.proto index 1be0acd..24266cd 100644 --- a/proto/transaction.proto +++ b/proto/transaction.proto @@ -5,14 +5,13 @@ package proto; option go_package = "github.com/babylonlabs-io/btc-staker/proto"; enum TransactionState { - TRANSACTION_CREATED = 0; - SENT_TO_BTC = 1; - CONFIRMED_ON_BTC = 2; - SENT_TO_BABYLON = 3; - VERIFIED = 4; - DELEGATION_ACTIVE = 5; - UNBONDING_CONFIRMED_ON_BTC = 6; - SPENT_ON_BTC = 7; + SENT_TO_BTC = 0; + CONFIRMED_ON_BTC = 1; + SENT_TO_BABYLON = 2; + VERIFIED = 3; + DELEGATION_ACTIVE = 4; + UNBONDING_CONFIRMED_ON_BTC = 5; + SPENT_ON_BTC = 6; } message WatchedTxData { diff --git a/staker/stakerapp.go b/staker/stakerapp.go index 3813bb1..5e4787e 100644 --- a/staker/stakerapp.go +++ b/staker/stakerapp.go @@ -527,14 +527,12 @@ func (app *StakerApp) checkTransactionsStatus() error { // Keep track of all staking transactions which need checking. chainhash.Hash objects are not relativly small // so it should not OOM even for larage database - var transactionCreated []*chainhash.Hash var transactionsSentToBtc []*chainhash.Hash var transactionConfirmedOnBtc []*chainhash.Hash var transactionsOnBabylon []*stakingDbInfo var transactionsVerifiedOnBabylon []*chainhash.Hash reset := func() { - transactionCreated = make([]*chainhash.Hash, 0) transactionsSentToBtc = make([]*chainhash.Hash, 0) transactionConfirmedOnBtc = make([]*chainhash.Hash, 0) transactionsOnBabylon = make([]*stakingDbInfo, 0) @@ -550,9 +548,6 @@ func (app *StakerApp) checkTransactionsStatus() error { // restarts stakingTxHash := tx.StakingTx.TxHash() switch tx.State { - case proto.TransactionState_TRANSACTION_CREATED: - transactionCreated = append(transactionCreated, &stakingTxHash) - return nil case proto.TransactionState_SENT_TO_BTC: transactionsSentToBtc = append(transactionsSentToBtc, &stakingTxHash) return nil @@ -598,82 +593,12 @@ func (app *StakerApp) checkTransactionsStatus() error { } app.logger.WithFields(logrus.Fields{ - "num_created": len(transactionCreated), "num_sent_to_btc": len(transactionsSentToBtc), "num_confirmed_on_btc": len(transactionConfirmedOnBtc), "num_on_babylon": len(transactionsOnBabylon), "num_verified": len(transactionsVerifiedOnBabylon), }).Debug("Iteration over all database staking requests finished") - for _, txHash := range transactionCreated { - txHashCopy := txHash - tx, stakerAddress := app.mustGetTransactionAndStakerAddress(txHashCopy) - - alreadyDelegated, err := app.babylonClient.IsTxAlreadyPartOfDelegation(txHashCopy) - - if err != nil { - // we got some communication err, return error and kill app startup - return err - } - - _, status, err := app.wc.TxDetails(txHashCopy, tx.StakingTx.TxOut[tx.StakingOutputIndex].PkScript) - - if err != nil { - // we got some communication err, return error and kill app startup - return err - } - - // transaction: - // - in created state - // - on babylon - // - not on btc chain - // resume pre-approval flow - if alreadyDelegated { - app.wg.Add(1) - go app.activateVerifiedDelegation( - tx.StakingTx, - tx.StakingOutputIndex, - txHashCopy, - ) - continue - } - - // transaction - // - not on babylon - // - not on btc chain - // - in created state - // resume pre-approval flow - if status == walletcontroller.TxNotFound { - req := &sendDelegationRequest{ - txHash: *txHashCopy, - inclusionInfo: nil, - requiredInclusionBlockDepth: stakingParams.ConfirmationTimeBlocks, - } - - app.wg.Add(1) - go app.sendDelegationToBabylonTask(req, stakerAddress, tx) - continue - } - - // transaction - // - not on babylon - // - on btc chain - // - in created state - // resume post-approval flow - if err := app.waitForStakingTransactionConfirmation( - txHashCopy, - tx.StakingTx.TxOut[tx.StakingOutputIndex].PkScript, - stakingParams.ConfirmationTimeBlocks, - app.currentBestBlockHeight.Load(), - ); err != nil { - return err - } - } - - app.logger.WithFields(logrus.Fields{ - "state": proto.TransactionState_TRANSACTION_CREATED.String(), - }).Debug("Partially fixed state of the database") - for _, txHash := range transactionsSentToBtc { stakingTxHash := txHash tx, _ := app.mustGetTransactionAndStakerAddress(stakingTxHash) @@ -700,7 +625,7 @@ func (app *StakerApp) checkTransactionsStatus() error { delegationInfo, err := app.babylonClient.QueryDelegationInfo(stakingTxHash) - if err != nil && !errors.Is(cl.ErrDelegationNotFound, err) { + if err != nil && !errors.Is(err, cl.ErrDelegationNotFound) { return err } diff --git a/stakerdb/trackedtranactionstore.go b/stakerdb/trackedtranactionstore.go index bad1943..da21bb2 100644 --- a/stakerdb/trackedtranactionstore.go +++ b/stakerdb/trackedtranactionstore.go @@ -587,7 +587,7 @@ func CreateTrackedTransaction( StakingTxBtcConfirmationInfo: nil, BtcSigType: pop.BtcSigType, BtcSigOverBbnStakerAddr: pop.BtcSigOverBabylonAddr, - State: proto.TransactionState_TRANSACTION_CREATED, + State: proto.TransactionState_SENT_TO_BTC, Watched: false, UnbondingTxData: nil, } From 82e0aba3e0d79b9229d02cf098e956212e04e6e4 Mon Sep 17 00:00:00 2001 From: KonradStaniec Date: Tue, 15 Oct 2024 17:22:37 +0200 Subject: [PATCH 03/11] Fix: do not send signed staking transaction to btc --- staker/stakerapp.go | 23 ++++++++++++++++++++--- stakerdb/trackedtransactionstore_test.go | 2 +- walletcontroller/interface.go | 2 +- 3 files changed, 22 insertions(+), 5 deletions(-) diff --git a/staker/stakerapp.go b/staker/stakerapp.go index 5e4787e..d232079 100644 --- a/staker/stakerapp.go +++ b/staker/stakerapp.go @@ -1316,7 +1316,23 @@ func (app *StakerApp) handlePreApprovalCmd(cmd *stakingRequestCmd) error { func (app *StakerApp) handlePostApprovalCmd(cmd *stakingRequestCmd) error { bestBlockHeight := app.currentBestBlockHeight.Load() - _, err := app.wc.SendRawTransaction(cmd.stakingTx, true) + err := app.wc.UnlockWallet(defaultWalletUnlockTimeout) + + if err != nil { + return err + } + + tx, fullySignd, err := app.wc.SignRawTransaction(cmd.stakingTx) + + if err != nil { + return err + } + + if !fullySignd { + return fmt.Errorf("failed to fully sign transaction with hash %s", cmd.stakingTxHash) + } + + _, err = app.wc.SendRawTransaction(tx, true) if err != nil { return err @@ -1780,8 +1796,9 @@ func (app *StakerApp) StakeFunds( feeRate := app.feeEstimator.EstimateFeePerKb() - // Create unsigned transaction by wallet - tx, err := app.wc.CreateAndSignTx([]*wire.TxOut{stakingInfo.StakingOutput}, btcutil.Amount(feeRate), stakerAddress) + // Create unsigned transaction by wallet without signing. Signing will happen + // in next steps + tx, err := app.wc.CreateTransaction([]*wire.TxOut{stakingInfo.StakingOutput}, btcutil.Amount(feeRate), stakerAddress) if err != nil { return nil, err diff --git a/stakerdb/trackedtransactionstore_test.go b/stakerdb/trackedtransactionstore_test.go index a1b4016..a22544d 100644 --- a/stakerdb/trackedtransactionstore_test.go +++ b/stakerdb/trackedtransactionstore_test.go @@ -195,7 +195,7 @@ func TestStateTransitions(t *testing.T) { // Inital state storedTx, err := s.GetTransaction(&txHash) require.NoError(t, err) - require.Equal(t, proto.TransactionState_TRANSACTION_CREATED, storedTx.State) + require.Equal(t, proto.TransactionState_SENT_TO_BTC, storedTx.State) require.Equal(t, uint64(1), storedTx.StoredTransactionIdx) // Confirmed hash := datagen.GenRandomBtcdHash(r) diff --git a/walletcontroller/interface.go b/walletcontroller/interface.go index 8dd4f4c..2a776ba 100644 --- a/walletcontroller/interface.go +++ b/walletcontroller/interface.go @@ -49,7 +49,7 @@ type WalletController interface { SignRawTransaction(tx *wire.MsgTx) (*wire.MsgTx, bool, error) // requires wallet to be unlocked CreateAndSignTx( - output []*wire.TxOut, + outputs []*wire.TxOut, feeRatePerKb btcutil.Amount, changeAddress btcutil.Address, ) (*wire.MsgTx, error) From abbb6745947a67d01c7c3bdfc7d08165ec00aebf Mon Sep 17 00:00:00 2001 From: KonradStaniec Date: Tue, 15 Oct 2024 17:27:39 +0200 Subject: [PATCH 04/11] change log entry --- CHANGELOG.md | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 13c979b..b52a157 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -37,9 +37,12 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) ### Misc Improvements -* [51](https://github.com/babylonlabs-io/btc-staker/pull/51) Use int64 +* [#51](https://github.com/babylonlabs-io/btc-staker/pull/51) Use int64 for satoshi amount related values. +* [#65](https://github.com/babylonlabs-io/btc-staker/pull/65) Various fixes to +pre-approval flow. Do not send signed staking transactions to Babylon. + ## v0.7.2 ### Bug fix From a7fdd7a5238112ce0fc263abe708e69a179a871f Mon Sep 17 00:00:00 2001 From: KonradStaniec Date: Tue, 15 Oct 2024 17:28:17 +0200 Subject: [PATCH 05/11] add mising commands file --- staker/commands.go | 130 +++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 130 insertions(+) create mode 100644 staker/commands.go diff --git a/staker/commands.go b/staker/commands.go new file mode 100644 index 0000000..2645a12 --- /dev/null +++ b/staker/commands.go @@ -0,0 +1,130 @@ +package staker + +import ( + cl "github.com/babylonlabs-io/btc-staker/babylonclient" + "github.com/btcsuite/btcd/btcec/v2" + "github.com/btcsuite/btcd/btcec/v2/schnorr" + "github.com/btcsuite/btcd/btcutil" + "github.com/btcsuite/btcd/chaincfg/chainhash" + "github.com/btcsuite/btcd/wire" + sdk "github.com/cosmos/cosmos-sdk/types" +) + +// we can make command to implement StakingEvent interface +var _ StakingEvent = (*stakingRequestCmd)(nil) + +type stakingRequestCmd struct { + stakerAddress btcutil.Address + stakingTxHash chainhash.Hash + stakingTx *wire.MsgTx + stakingOutputIdx uint32 + stakingOutputPkScript []byte + stakingTime uint16 + stakingValue btcutil.Amount + fpBtcPks []*btcec.PublicKey + requiredDepthOnBtcChain uint32 + pop *cl.BabylonPop + watchTxData *watchTxDataCmd + usePreApprovalFlow bool + errChan chan error + successChan chan *chainhash.Hash +} + +func (req *stakingRequestCmd) isWatched() bool { + return req.watchTxData != nil +} + +func newOwnedStakingCommand( + stakerAddress btcutil.Address, + stakingTx *wire.MsgTx, + stakingOutputIdx uint32, + stakingOutputPkScript []byte, + stakingTime uint16, + stakingValue btcutil.Amount, + fpBtcPks []*btcec.PublicKey, + confirmationTimeBlocks uint32, + pop *cl.BabylonPop, + usePreApprovalFlow bool, +) *stakingRequestCmd { + return &stakingRequestCmd{ + stakerAddress: stakerAddress, + stakingTxHash: stakingTx.TxHash(), + stakingTx: stakingTx, + stakingOutputIdx: stakingOutputIdx, + stakingOutputPkScript: stakingOutputPkScript, + stakingTime: stakingTime, + stakingValue: stakingValue, + fpBtcPks: fpBtcPks, + requiredDepthOnBtcChain: confirmationTimeBlocks, + pop: pop, + watchTxData: nil, + usePreApprovalFlow: usePreApprovalFlow, + errChan: make(chan error, 1), + successChan: make(chan *chainhash.Hash, 1), + } +} + +type watchTxDataCmd struct { + slashingTx *wire.MsgTx + slashingTxSig *schnorr.Signature + stakerBabylonAddr sdk.AccAddress + stakerBtcPk *btcec.PublicKey + // unbonding related data + unbondingTx *wire.MsgTx + slashUnbondingTx *wire.MsgTx + slashUnbondingTxSig *schnorr.Signature + unbondingTime uint16 +} + +func newWatchedStakingCmd( + stakerAddress btcutil.Address, + stakingTx *wire.MsgTx, + stakingOutputIdx uint32, + stakingOutputPkScript []byte, + stakingTime uint16, + stakingValue btcutil.Amount, + fpBtcPks []*btcec.PublicKey, + confirmationTimeBlocks uint32, + pop *cl.BabylonPop, + slashingTx *wire.MsgTx, + slashingTxSignature *schnorr.Signature, + stakerBabylonAddr sdk.AccAddress, + stakerBtcPk *btcec.PublicKey, + unbondingTx *wire.MsgTx, + slashUnbondingTx *wire.MsgTx, + slashUnbondingTxSig *schnorr.Signature, + unbondingTime uint16, +) *stakingRequestCmd { + return &stakingRequestCmd{ + stakerAddress: stakerAddress, + stakingTxHash: stakingTx.TxHash(), + stakingTx: stakingTx, + stakingOutputIdx: stakingOutputIdx, + stakingOutputPkScript: stakingOutputPkScript, + stakingTime: stakingTime, + stakingValue: stakingValue, + fpBtcPks: fpBtcPks, + requiredDepthOnBtcChain: confirmationTimeBlocks, + pop: pop, + watchTxData: &watchTxDataCmd{ + slashingTx: slashingTx, + slashingTxSig: slashingTxSignature, + stakerBabylonAddr: stakerBabylonAddr, + stakerBtcPk: stakerBtcPk, + unbondingTx: unbondingTx, + slashUnbondingTx: slashUnbondingTx, + slashUnbondingTxSig: slashUnbondingTxSig, + unbondingTime: unbondingTime, + }, + errChan: make(chan error, 1), + successChan: make(chan *chainhash.Hash, 1), + } +} + +func (event *stakingRequestCmd) EventId() chainhash.Hash { + return event.stakingTxHash +} + +func (event *stakingRequestCmd) EventDesc() string { + return "STAKING_REQUESTED_CMD" +} From 49640acf6f7f9fed85f12f52581d15feb51bd349 Mon Sep 17 00:00:00 2001 From: KonradStaniec Date: Tue, 15 Oct 2024 17:30:35 +0200 Subject: [PATCH 06/11] fix linter --- staker/events.go | 20 -------------------- 1 file changed, 20 deletions(-) diff --git a/staker/events.go b/staker/events.go index b171e52..9485f21 100644 --- a/staker/events.go +++ b/staker/events.go @@ -7,11 +7,6 @@ import ( "github.com/sirupsen/logrus" ) -type responseExpectedChan struct { - errChan chan error - successChan chan *chainhash.Hash -} - type StakingEvent interface { // Each staking event is identified by initial staking transaction hash EventId() chainhash.Hash @@ -24,7 +19,6 @@ var _ StakingEvent = (*delegationActiveOnBabylonEvent)(nil) var _ StakingEvent = (*unbondingTxSignaturesConfirmedOnBabylonEvent)(nil) var _ StakingEvent = (*unbondingTxConfirmedOnBtcEvent)(nil) var _ StakingEvent = (*spendStakeTxConfirmedOnBtcEvent)(nil) -var _ StakingEvent = (*sendStakingTxToBTCRequestedEvent)(nil) var _ StakingEvent = (*criticalErrorEvent)(nil) type stakingTxBtcConfirmedEvent struct { @@ -127,20 +121,6 @@ func (app *StakerApp) logStakingEventProcessed(event StakingEvent) { }).Debug("Processed staking event") } -type sendStakingTxToBTCRequestedEvent struct { - stakingTxHash chainhash.Hash - requiredDepthOnBtcChain uint32 - responseExpected *responseExpectedChan -} - -func (event *sendStakingTxToBTCRequestedEvent) EventId() chainhash.Hash { - return event.stakingTxHash -} - -func (event *sendStakingTxToBTCRequestedEvent) EventDesc() string { - return "SEND_STAKING_TX_TO_BTC_REQUESTED" -} - type delegationActiveOnBabylonEvent struct { stakingTxHash chainhash.Hash } From 419cb1ebb5959fde96ab993a27ecad85f3227c3d Mon Sep 17 00:00:00 2001 From: RafilxTenfen Date: Tue, 15 Oct 2024 20:12:36 -0300 Subject: [PATCH 07/11] chore: update babylon to branch rafilx/btc-staking-activation-height --- go.mod | 2 +- go.sum | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/go.mod b/go.mod index e87b4d7..b86f8b2 100644 --- a/go.mod +++ b/go.mod @@ -6,7 +6,7 @@ require ( cosmossdk.io/errors v1.0.1 cosmossdk.io/math v1.3.0 github.com/avast/retry-go/v4 v4.5.1 - github.com/babylonlabs-io/babylon v0.12.0 + github.com/babylonlabs-io/babylon v0.9.3-0.20241015230429-26632eaa78a0 github.com/babylonlabs-io/networks/parameters v0.2.2 github.com/btcsuite/btcd v0.24.2 github.com/btcsuite/btcd/btcec/v2 v2.3.2 diff --git a/go.sum b/go.sum index 61c9a22..8a1f6fd 100644 --- a/go.sum +++ b/go.sum @@ -287,8 +287,8 @@ github.com/aws/aws-sdk-go v1.44.122/go.mod h1:y4AeaBuwd2Lk+GepC1E9v0qOiTws0MIWAX github.com/aws/aws-sdk-go v1.44.312 h1:llrElfzeqG/YOLFFKjg1xNpZCFJ2xraIi3PqSuP+95k= github.com/aws/aws-sdk-go v1.44.312/go.mod h1:aVsgQcEevwlmQ7qHE9I3h+dtQgpqhFB+i8Phjh7fkwI= github.com/aws/aws-sdk-go-v2 v0.18.0/go.mod h1:JWVYvqSMppoMJC0x5wdwiImzgXTI9FuZwxzkQq9wy+g= -github.com/babylonlabs-io/babylon v0.12.0 h1:s2OTcxpk0RzrkVBnVTfnPdJVYDSqnm/33YKPQqEzNCE= -github.com/babylonlabs-io/babylon v0.12.0/go.mod h1:ZOrTde9vs2xoqGTFw4xhupu2CMulnpywiuk0eh4kPOw= +github.com/babylonlabs-io/babylon v0.9.3-0.20241015230429-26632eaa78a0 h1:s/U5FjGSQQwArPI2iYXcnMJvVfww+8y14pjmx6akAak= +github.com/babylonlabs-io/babylon v0.9.3-0.20241015230429-26632eaa78a0/go.mod h1:cxRwVqVLoJ39FpyovTEHJLu1lwwrM1tE8davu7nRHwY= github.com/babylonlabs-io/networks/parameters v0.2.2 h1:TCu39fZvjX5f6ZZrjhYe54M6wWxglNewuKu56yE+zrc= github.com/babylonlabs-io/networks/parameters v0.2.2/go.mod h1:iEJVOzaLsE33vpP7J4u+CRGfkSIfErUAwRmgCFCBpyI= github.com/benbjohnson/clock v1.1.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA= From a704f9791da4f1525f5a4fc5216ad9b8917c33b2 Mon Sep 17 00:00:00 2001 From: RafilxTenfen Date: Tue, 15 Oct 2024 20:16:53 -0300 Subject: [PATCH 08/11] chore: update babylon new funciton names --- babylonclient/babyloncontroller.go | 6 +++--- babylonclient/interface.go | 6 +++--- staker/types.go | 4 ++-- 3 files changed, 8 insertions(+), 8 deletions(-) diff --git a/babylonclient/babyloncontroller.go b/babylonclient/babyloncontroller.go index 7a9a625..59c33bc 100644 --- a/babylonclient/babyloncontroller.go +++ b/babylonclient/babyloncontroller.go @@ -171,9 +171,9 @@ func (bc *BabylonController) Params() (*StakingParams, error) { return nil, err } - minUnbondingTimeU64 := sdkmath.Max[uint64]( + minUnbondingTimeU64 := sdkmath.Max[uint32]( bccParams.CheckpointFinalizationTimeout, - uint64(stakingTrackerParams.MinUnbondingTime), + uint32(stakingTrackerParams.MinUnbondingTime), ) if minUnbondingTimeU64 > math.MaxUint16 { @@ -663,7 +663,7 @@ func (bc *BabylonController) QueryFinalityProvider(btcPubKey *btcec.PublicKey) ( }, nil } -func (bc *BabylonController) QueryHeaderDepth(headerHash *chainhash.Hash) (uint64, error) { +func (bc *BabylonController) QueryHeaderDepth(headerHash *chainhash.Hash) (uint32, error) { ctx, cancel := getQueryContext(bc.cfg.Timeout) defer cancel() diff --git a/babylonclient/interface.go b/babylonclient/interface.go index 77e33a0..de6a6b9 100644 --- a/babylonclient/interface.go +++ b/babylonclient/interface.go @@ -70,7 +70,7 @@ type BabylonClient interface { Undelegate(req *UndelegationRequest) (*pv.RelayerTxResponse, error) QueryFinalityProviders(limit uint64, offset uint64) (*FinalityProvidersClientResponse, error) QueryFinalityProvider(btcPubKey *btcec.PublicKey) (*FinalityProviderClientResponse, error) - QueryHeaderDepth(headerHash *chainhash.Hash) (uint64, error) + QueryHeaderDepth(headerHash *chainhash.Hash) (uint32, error) IsTxAlreadyPartOfDelegation(stakingTxHash *chainhash.Hash) (bool, error) QueryDelegationInfo(stakingTxHash *chainhash.Hash) (*DelegationInfo, error) } @@ -142,9 +142,9 @@ func (m *MockBabylonClient) QueryFinalityProvider(btcPubKey *btcec.PublicKey) (* } } -func (m *MockBabylonClient) QueryHeaderDepth(headerHash *chainhash.Hash) (uint64, error) { +func (m *MockBabylonClient) QueryHeaderDepth(headerHash *chainhash.Hash) (uint32, error) { // return always confirmed depth - return uint64(m.ClientParams.ConfirmationTimeBlocks) + 1, nil + return m.ClientParams.ConfirmationTimeBlocks + 1, nil } func (m *MockBabylonClient) IsTxAlreadyPartOfDelegation(stakingTxHash *chainhash.Hash) (bool, error) { diff --git a/staker/types.go b/staker/types.go index 6a59991..f31db69 100644 --- a/staker/types.go +++ b/staker/types.go @@ -517,7 +517,7 @@ func parseWatchStakingRequest( } // 2. Check wheter slashing tx match staking tx - err = staking.CheckTransactions( + err = staking.CheckSlashingTxMatchFundingTx( slashingTx, stakingTx, stakingOutputIdx, @@ -584,7 +584,7 @@ func parseWatchStakingRequest( return nil, fmt.Errorf("failed to watch staking tx. Unbonding output does not match output produced from provided values") } - err = staking.CheckTransactions( + err = staking.CheckSlashingTxMatchFundingTx( slashUnbondingTx, unbondingTx, 0, From db58efa38ec67ce992852f1ef80921241c86e6bc Mon Sep 17 00:00:00 2001 From: RafilxTenfen Date: Wed, 16 Oct 2024 12:54:28 -0300 Subject: [PATCH 09/11] chore: update to babylon v013 --- go.mod | 2 +- go.sum | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/go.mod b/go.mod index b86f8b2..95e8fe4 100644 --- a/go.mod +++ b/go.mod @@ -6,7 +6,7 @@ require ( cosmossdk.io/errors v1.0.1 cosmossdk.io/math v1.3.0 github.com/avast/retry-go/v4 v4.5.1 - github.com/babylonlabs-io/babylon v0.9.3-0.20241015230429-26632eaa78a0 + github.com/babylonlabs-io/babylon v0.13.0 github.com/babylonlabs-io/networks/parameters v0.2.2 github.com/btcsuite/btcd v0.24.2 github.com/btcsuite/btcd/btcec/v2 v2.3.2 diff --git a/go.sum b/go.sum index 8a1f6fd..266f12b 100644 --- a/go.sum +++ b/go.sum @@ -287,8 +287,8 @@ github.com/aws/aws-sdk-go v1.44.122/go.mod h1:y4AeaBuwd2Lk+GepC1E9v0qOiTws0MIWAX github.com/aws/aws-sdk-go v1.44.312 h1:llrElfzeqG/YOLFFKjg1xNpZCFJ2xraIi3PqSuP+95k= github.com/aws/aws-sdk-go v1.44.312/go.mod h1:aVsgQcEevwlmQ7qHE9I3h+dtQgpqhFB+i8Phjh7fkwI= github.com/aws/aws-sdk-go-v2 v0.18.0/go.mod h1:JWVYvqSMppoMJC0x5wdwiImzgXTI9FuZwxzkQq9wy+g= -github.com/babylonlabs-io/babylon v0.9.3-0.20241015230429-26632eaa78a0 h1:s/U5FjGSQQwArPI2iYXcnMJvVfww+8y14pjmx6akAak= -github.com/babylonlabs-io/babylon v0.9.3-0.20241015230429-26632eaa78a0/go.mod h1:cxRwVqVLoJ39FpyovTEHJLu1lwwrM1tE8davu7nRHwY= +github.com/babylonlabs-io/babylon v0.13.0 h1:h7cazmFmItePvZHEbLhDbsq2l7xN4e2AjDHRM7zDLkg= +github.com/babylonlabs-io/babylon v0.13.0/go.mod h1:cxRwVqVLoJ39FpyovTEHJLu1lwwrM1tE8davu7nRHwY= github.com/babylonlabs-io/networks/parameters v0.2.2 h1:TCu39fZvjX5f6ZZrjhYe54M6wWxglNewuKu56yE+zrc= github.com/babylonlabs-io/networks/parameters v0.2.2/go.mod h1:iEJVOzaLsE33vpP7J4u+CRGfkSIfErUAwRmgCFCBpyI= github.com/benbjohnson/clock v1.1.0/go.mod h1:J11/hYXuz8f4ySSvYwY0FKfm+ezbsZBKZxNJlLklBHA= From 27d9fc4dc8878dc0cc9832e7c25a28865e04cbc7 Mon Sep 17 00:00:00 2001 From: RafilxTenfen Date: Wed, 16 Oct 2024 12:56:19 -0300 Subject: [PATCH 10/11] chore: add #66 to changelog --- CHANGELOG.md | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 8a74084..06b348f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -39,12 +39,12 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) * [#51](https://github.com/babylonlabs-io/btc-staker/pull/51) Use int64 for satoshi amount related values. - * [#65](https://github.com/babylonlabs-io/btc-staker/pull/65) Various fixes to pre-approval flow. Do not send signed staking transactions to Babylon. - +* [#66](https://github.com/babylonlabs-io/btc-staker/pull/66) Bump babylon to +v0.13.0. * [#67](https://github.com/babylonlabs-io/btc-staker/pull/67) Enable concurrent -sending of multiple pre-approval staking transactions +sending of multiple pre-approval staking transactions. ## v0.7.2 From 06835b7166c589d94402ad1abc3b2018601a4545 Mon Sep 17 00:00:00 2001 From: RafilxTenfen Date: Wed, 16 Oct 2024 12:58:14 -0300 Subject: [PATCH 11/11] chore: better changelog desc for #66 --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 06b348f..8d35f4b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -42,7 +42,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) * [#65](https://github.com/babylonlabs-io/btc-staker/pull/65) Various fixes to pre-approval flow. Do not send signed staking transactions to Babylon. * [#66](https://github.com/babylonlabs-io/btc-staker/pull/66) Bump babylon to -v0.13.0. +v0.13.0 as it updates BTC block heights to uint32 and function naming. * [#67](https://github.com/babylonlabs-io/btc-staker/pull/67) Enable concurrent sending of multiple pre-approval staking transactions.