Staking Transactions
Specification of all the transactions, which is for those who want to construct Bitcoin staking related transactions by themselves instead of using our code or dapps.
Introduction
A lock-only network involves users locking their Bitcoin using the self-custodial Bitcoin Staking script without a Babylon Genesis operating. In this document, we precisely define how one can construct the Bitcoin transactions specified by the Bitcoin Staking protocol.
Prerequisites
- Scripts doc - a document which defines how different Bitcoin Staking scripts look like
- BIP341 - a document specifying how to spend Taproot outputs
System parameters
The lock-only staking system is governed by a set of parameters that specify
what constitutes a valid staking transaction. Based on those,
an observer of the Bitcoin ledger can precisely identify which transactions
are valid staking transactions and whether they should be considered active stake.
These parameters are different depending on the Bitcoin height a transaction is
included in and a constructor of a Bitcoin Staking transaction should take them into
account before propagating a transaction to Bitcoin.
For the rest of the document, we will refer to those parameters as global_parameters
.
More details about parameters can be found in the parameters spec.
Taproot outputs
Taproot outputs are outputs whose locking script is an elliptic curve point Q
created as follows:
Q = P + hash(P||m)G
where:
P
is the internal public keym
is the root of a Merkle tree whose leaves consist of a version number and a script
For Bitcoin Staking transactions, the internal public key is chosen as:
P = lift_x(0x50929b74c1a04954b78b4b6035e97a5e078a5a0f28ec96d547bfee9ace803ac0)
This key is described in the BIP341 specification.
The use of this key as an internal public key disables spending from Taproot output through the key spending path. The construction of this key can be found here.
Observable Staking Transactions
Staking transaction
A staker enters the system through the creation of a staking transaction which locks Bitcoin in the Bitcoin Staking script.
Requirements
For the transaction to be considered a valid staking transaction, it must:
- Have a Taproot output which has the key spending path disabled
and commits to a script tree composed of three scripts:
timelock script, unbonding script, slashing script.
This output is henceforth known as the
staking_output
and the value in this output is known asstaking_amount
- Have
OP_RETURN
output which contains:global_parameters.tag
,version
,staker_pk
,finality_provider_pk
,staking_time
- All the values must be valid for the
global_parameters
which are applicable at the height in which the staking transaction is included in the BTC ledger.
OP_RETURN output description
Data in the OP_RETURN output is described by the following struct:
type V0OpReturnData struct {
Tag []byte
Version byte
StakerPublicKey []byte
FinalityProviderPublicKey []byte
StakingTime []byte
}
The implementation of the struct can be found here
Fields description:
Tag
- 4 bytes, a tag which is used to identify the staking transaction among other transactions in the Bitcoin ledger. It is specified in theglobal_parameters.Tag
field.Version
- 1 byte, the current version of the OP_RETURN output.StakerPublicKey
- 32 bytes, staker public key. The same key must be used in the scripts used to create the Taproot output in the staking transaction.FinalityProviderPublicKey
- 32 bytes, finality provider public key. The same key must be used in the scripts used to create the Taproot output in the staking transaction.StakingTime
- 2 bytes big-endian unsigned number, staking time. The same timelock time must be used in scripts used to create the Taproot output in the staking transaction.
This data is serialized as follows:
SerializedStakingData = Tag || Version || StakerPublicKey || FinalityProviderPublicKey || StakingTime
To transform this data into OP_RETURN data:
StakingDataPkScript = 0x6a || 0x47 || SerializedStakingData
where:
- 0x6a - is byte marker representing OP_RETURN op code.
- 0x47 - is byte marker representing OP_DATA_71 op code, which pushed 71 bytes onto the stack.
The final OP_RETURN output will have the following shape:
TxOut {
Value: 0,
PkScript: StakingDataPkScript
}
Logic creating output from data can be found here
Staking output description
Staking output should commit to three scripts:
timelock_script
unbonding_script
slashing_script
Data needed to create staking_output
:
staker_public_key
- chosen by the user sending the staking transaction. It will be used in every script. This key needs to be put in the OP_RETURN output in the staking transaction.finality_provider_public_key
- chosen by the user sending the staking transaction. It will be used as<FinalityPk>
in theslashing_script
. In the lock-only network, there is no slashing, so this key has mostly informative purposes. This key needs to be put in the OP_RETURN output of the staking transaction.staking_time
- chosen by the user sending the staking transaction. It will be used as the locking time in thetimelock_script
. It must be a validuint16
number, in the rangeglobal_parameters.min_staking_time <= staking_time <= global_parameters.max_staking_time
. It needs to be put in the OP_RETURN output of the staking transaction.covenant_committee_public_keys
- it can be retrieved fromglobal_parameters.covenant_pks
. It is a set of covenant committee public keys which will be put inunbonding_script
andslashing_script
.covenant_committee_quorum
- it can be retrieved fromglobal_parameters.covenant_quorum
. It is the quorum of covenant committee members required to authorize spending using theunbonding_script
orslashing_script
.staking_amount
- chosen by the user, it will be placed instaking_output.value
.btc_network
- the BTC network on which staking transactions will take place.
Building OP_RETURN and staking output implementation
The Babylon staking library exposes the BuildV0IdentifiableStakingOutputsAndTx function with the following signature:
func BuildV0IdentifiableStakingOutputsAndTx(
tag []byte,
stakerKey *btcec.PublicKey,
fpKey *btcec.PublicKey,
covenantKeys []*btcec.PublicKey,
covenantQuorum uint32,
stakingTime uint16,
stakingAmount btcutil.Amount,
net *chaincfg.Params,
) (*IdentifiableStakingInfo, *wire.MsgTx, error)
It enables the caller to create valid outputs to put inside an unfunded and not-signed staking transaction.
The suggested way of creating and sending a staking transaction using bitcoind is:
- Create
staker_key
in the bitcoind wallet. - Create unfunded and not signed staking transaction using
the
BuildV0IdentifiableStakingOutputsAndTx
function. - Serialize the unfunded and not signed staking transaction to
staking_transaction_hex
. - Call
bitcoin-cli fundrawtransaction "staking_transaction_hex"
to retrievefunded_staking_transaction_hex
. The bitcoind wallet will automatically choose unspent outputs to fund this transaction. - Call
bitcoin-cli signrawtransactionwithwallet "funded_staking_transaction_hex"
. This call will sign all inputs of the transaction and returnsigned_staking_transaction_hex
. - Call
bitcoin-cli sendrawtransaction "signed_staking_transaction_hex"
.
Unbonding transaction
The unbonding transaction allows the staker to on-demand unbond their locked Bitcoin stake prior to its original timelock expiration.