Vega runs on a delegated proof of stake (DPOS) blockchain. Participants who hold a balance of the configured governance asset can stake these on the network by delegating their tokens to one or more validators that they trust. This helps to secure the network.
Validators and delegators receive incentives from the network, depending on various factors, including how much stake is delegated and how honest they are.
Staking requires the combined action of:
- Associating tokens on the Vega staking bridge contract; and
- Nominating these tokens to one or more validators
- Delegation in some contexts is used to mean
associate + nominate
. For the purposes of this document, once it's clear from context that association has happeneddelegate
andnominate
may be used interchangeably.
Delegation and staking are terms that may be used interchangeably, since delegation is the act of staking VEGA tokens on a validator. A delegator can associate a token in the Vega staking bridge contract, which is then available for nomination. To this end, a Vega token (or a fraction thereof) can be:
- Unassociated: The tokenholder is free to do with the token as they want, but cannot nominate it
- Associated: The token is locked in the staking and delegation smart contract and associated to a Vega key. It can be used on the Vega chain for governance and it can be nominated to a validator.
It is important that no action triggered on Vega needs to directly invoke the Vega staking bridge contract through the validators; thus, all actions regarding associating and dissociating of stake are initiated by the Vega staking bridge contract, not by the Vega chain.
In order to delegate, users require tokens that will be associated in a smart contract (see Vega staking bridge contract). Vega will be made aware of how many tokens a given party has associated through bridge events. When the same tokens are dissociated, a corresponding event will be emitted:
Note that the bridge contract uses deposited
and removed
instead of associated
and dissociated
.
event Stake_Deposited(address indexed user, uint256 amount, bytes32 vega_public_key);
event Stake_Removed(address indexed user, uint256 amount);
This provides the information the core needs to keep track of:
- Total delegateable Stake
- Undelegated Stake
- Stake delegated per validator
- Stake marked for delegation per validator in the next epoch.
- Total stake (should be the sum of all those listed immediately above).
There is no interaction with the smart contract that is initiated by Vega.
The validators watch for events emitted by the staking and delegation smart contract, and observe the following actions:
This token is now available for delegation.
If the token holder has sufficient undelegated tokens, these are used to cover this request (i.e., the available amount of delegateable tokens is reduced to match the (un)locking status).
This could mean that the token-holder has a delegation-command scheduled that is no longer executable; this command will then be ignored at the start of the next epoch.
If the token holder does not have sufficient undelegated stake, at first the validators verify if tokens are in the process of being delegated (i.e., the delegation command has been issued, but not yet executed), and uses those tokens to cover the unlocking. If this is insufficient, the undelegate-now
command is automatically triggered, undelegating evenly from all validators to cover the needs.
Any locked and undelegated stake can be delegated at any time by putting a delegation-message on the chain. However, the delegation only becomes valid towards the next epoch, though it can be undone through undelegate.
Once Vega is aware of locked tokens, the users will have an account with the balance reflecting how many tokens were locked. At this point, the user can submit a transaction to stake (delegate) their tokens. The amount they stake must be <= balance
, naturally.
message Delegate {
uint256 Amount = 1;
Validator Val = 2;
}
message Undelegate {
uint256 Amount = 1;
Validator Val = 2;
}
Where Delegate
adds the Amount
to the delegation of validator Val
at the beginning of the next epoch (if still available to them), and Undelegate
subtracts the amount from the delegation of Val
by the next epoch if available.
To avoid fragmentation or spam, there is a network parameter minimum delegateable stake
that defines the smallest unit of (fractions of) tokens that can be used for delegation.
To delegate stake, a delegator simply puts a command "delegate x stake to y" on the chain. It is verified at the beginning (when the command is issued and before it is put on the chain) that the delegator has sufficient unlocked stake, as well as in the beginning of the next epoch just before the command takes effect. The amount of delegateable stake is reduced right away once the command is put into a block.
There is not maximum amount of stake that a validator can accept, instead at the end of the epoch when staking rewards are calculated, the stake of each validator (and their delegator) may be penalised if it represents a stake that it more than the optimal stake, i.e. the desired stake to be owned by each validator and its delegators.
Users can remove stake by submitting an Undelegate
transaction. The tokens will then be restored back to their token balance.
At the top level, Stake_Deposited
simply adds amount
of tokens to the account of the user associated with the user
. Likewise, the Stake_Removed
event subtracts the amount
of tokens from their account.
- If the
Stake_Removed
amount of tokens is higher than the balance of said user, something went seriously wrong somewhere. This is a good time to panic. - If the amount is higher than the amount of undelegated stake, the missing amount must be freed using the undelegate function (see section above about bridge contract interaction). There is currently no rule how to choose this;
Option-1 A first heuristic would be to take from the highest delegation first and then go down, e.g.
- If the delegation is 100, 90, 80, 70, and we need to free 30 stake, we first take from the richest ones until they are no longer the richest:
- Free 10, delegation is 90, 90, 80, 70
- Free 30, delegation is 80, 80, 80, 70
This has the benefit of lowering the probability that a single withdrawal will leave any one validator with zero delegated stake.
Option-2 Another option would be to withdraw stake proportionally from the validators.
- If the delegation is 100, 90, 80, 70, and we need to free 30 stake, we split the withdrawal across all validators proportionately:
- Free from delegator-1 (to whom the participant has delegated 100) an amount equal to 30 * (100/(100+90+80+70)) etc. Not sure how to deal with rounding.
- The action is announced in the next available block, but the delegator keeps the delegation alive till the last block of the epoch. The delegator can then re-delegate the stake, which then be valid once the next epoch starts. The delegator cannot move the tokens before the epoch ends, they remain locked.
- The action can be announced at any time and is executed immediately following the block it is announced in.
- The user is marked to not receive any reward from the validator in that epoch. - The reward should instead go into the on-chain treasury account for that asset. The stake is marked as free for the delegator, but is not yet removed from the validator stake (this happens at the end of the epoch).
Rationale: This allows a delegator to sell their tokens in a rush, without requiring any interaction between the smart contract and the details of the delegation system. This also allows the delegator to change their mind about a delegation before it is activated.
- This is not strictly a type of undelegation but it's effect is quite similar to undelegate now. This is expressed by unstaking rather than by un-delegating. When removing the stake, the corresponding stake will automatically be undelegated.
- A party become eligible to participate in auto delegation once they have manually delegated (nominated) over x% of the association. In theory this should be 100% but in practice due to rounding issues we can make this closer to 100%. It is currently defined as 95% of the association.
- Once entering auto delegation mode, any un-nominated associated tokens will be automatically distributed according to the current validator nomination of the party maintaining the same proportion.
- Edge cases:
- If a party has entered auto delegation mode, and their association has increased it should be automatically distributed for the epoch following the increase of association. However, if during the same epoch the party requests to execute manual delegation, no automatic delegation will be done in that epoch. If there is still un-nominated association in the next epoch, it will be automatically distributed.
- If a party qualifies for auto delegation and have un-nominated association, however the party requests to undelegate (either during the epoch or at the end of the epoch) - they exit auto delegation mode. The rationale here is that they probably want to do some rearrangement of their nomination and we give them a chance to do so. Once the party reached more than x% of nomination again, they would enter auto delegation mode again and any future un-nominated association will be automatically distributed.
- When distributing the newly available association according to the current validators nomination of the party, if validator A should get X but can only accept X - e (due to max per validator constraint), we don't try to distribute e between the other validators and will try to distribute it again in the next round.
- Auto undelegation - whenever the party dissociates tokens, their nomination must be updated such that their maximum nomination reflects the association.
A delegator can delegate some stake, and immediately undelegate it before the next epoch starts. This is fine with us.
If the value of minimum_delegateable_stake
changes in a bad way, stakers might be stuck with
some fraction they can't modify anymore. To this end, the undelegate commands also should
support a parameter "all".
With this setup, a delegator can use a constant delegation/undelegate-now to spam the network.
If several delegators change the delegation within the same block, some of them may not be allowed to execute (as this would exceed the maximum stake the validator wants). To save resources, the block creator has the responsibility to filter out these transactions.
Another edge case is the following: during the epoch the party had x tokens associated and they requested to nominate no validators 1-5 each x/5. Before the end of the epoch the party withdraws some of the association leaving insufficient to cover all of the nominations. In such a case the nominations are adjusted proportionally to the requests against the available association balance. For example, suppose the party had 500 tokens associated and they requested to nominate 100 to each of validators 1-5. Before the epoch ends the party dissociates 400 leaving only 100 tokens available. In this case each validator would get a nomination of 100/5=20. To be more accurate the way this works is as follows: for each of the validators we calculate first how much of the nomination requested would actually go through, e.g. if the request is for a 100 but the validator would only accept 20, then the effective amount considered is 20. Then we normalise the effective account (divide by total) and apply this factor on the available balance. The sum of these nominations is guaranteed to be less than or equal to the available un-nominated association.
Property | Type | Example value | Description |
---|---|---|---|
validators.delegation.minAmount |
String (float) | "0.001" |
The smallest fraction of the governance token that can be delegated to a validator. |
Actual validator score calculation is in [simple scheme for Sweetwater](0061-simple-POS-rewards\ -\ SweetWater.md) and it introduces its own network parameters.
See the network parameters spec for a full list of parameters.
To lock tokens, a participant must:
- Have some balance of vested or unvested governance asset in an Ethereum wallet. These assets must not be locked to another smart contract (including the Vega collateral bridge).
- Have a Vega wallet
- Lock the tokens on the Vega staking bridge contract,
To delegate the locked tokens, a participant must:
- Have enough tokens to satisfy the network parameter: "Minimum delegateable stake" (0059-STKG-001)
- Delegate the locked tokens to one of the eligible validators (fixed set for Alpha mainnet).(0059-STKG-002)
These accounts will be created:
- A staking account denominated in the governance asset is created(0059-STKG-003)
- When first fees are received as a staking reward, a general account for each settlement currency (so they can receive infrastructure fee rewards) (0059-STKG-004)
- It is possible that a separate reward function will cause an account to be created for the user as a result of rewards.
- Timings
- Any locked (but undelegated) tokens can be delegated at any time. (0059-STKG-005)
- The delegation only becomes valid at the next epoch, though it can be undone through undelegate. (0059-STKG-006)
- The balance of "delegateable stake" is reduced immediately (prior to it coming into effect in the next epoch) (0059-STKG-007)
- More tokens may be locked at any time on the Vega staking bridge contract (0059-STKG-008)
- More stake may be delegated at any time (see function: Stake - amount refers to size by which to increment existing staked amount) (0059-STKG-009)
- Same timings apply as per staking for the first time (0059-STKG-010)
- Any stake may be withdrawn from the Vega staking bridge contract at any time (0059-STKG-011)
- Unlocking your tokens in the bridge contract will effectively "remove" them from any delegation they're doing (unless you have remaining undelegated tokens that could fulfil your delegation) (0059-STKG-012)
- Delegation may be fully or partially removed. The amount specified in the function: Remove - is the size by which the existing staked amount will be decremented (0059-STKG-013)
- Removal of delegation may happen in the following 2 ways:
- Announcing removal, but maintaining stake until last block of the current epoch. This "announced stake" may be then (re)delegated (e.g. to a different validator). (0059-STKG-014)
- Announcing removal and withdrawing stake immediately. Rewards are still collected for this stake until the end of the epoch, but they are sent to the on-chain treasury account for that asset. (0059-STKG-015)
- Every 30 seconds (and at the end of an epoch) the associated stake is reconciled against the current nomination to ensure that the total nomination is not exceeding the total association. In case it does we proportionally un-nominate from the validators until the nomination is not exceeding the association. It's worth mentioning that for the current epoch nomination balance we consider the minimum stake that the party had for the duration of the period evaluated (from the beginning of the epoch). Also, for next epoch's nomination balance we consider the current balance of the association. Have one party stake 100 tokens split 70-30 between two validators. With a an epoch of length 5 minutes, at the beginning of the epoch remove 50 tokens off the stake. Within 30 seconds of the confirmation of updated association, verify that the nomination balances for the current epoch and next has been removed with respect to the 70/30 split, such that it is now 35/15. (0059-STKG-016)
- Changing the validator to whom a participant wants to validate to involves:
- Announcing removal of stake for current validator
- Staking on the new validator, as per normal function: Stake
- These can happen concurrently, so that at the next epoch, the stake is removed from the current validator and staked on the new validator
Normal scenario auto undelegation: (0059-STKG-018)
- epoch 0: party associated 1000 VEGA
- epoch 0: party nominated 200 VEGA to validators 1-5
- epoch 1: party dissociated 200 VEGA
- at the end of epoch 1:
party1
would have left 160 tokens nominated to validators 1-5 (for both epoch 1 and onwards - the former is important so that they don't get rewarded for 200 per validator)
Normal scenario auto delegation: (0059-STKG-019)
- epoch 0: party associated 1000 VEGA
- epoch 0: party nominated 200 VEGA to validators 1-5
- epoch 1: party associated 200 VEGA
- end of epoch 1: there's sufficient space on each validator 1-5 to accept the delegation of 40 VEGA from
party1
andparty1
now has delegation of 240 for validators 1-5 for epoch 2.
Edge case 1: manual delegation for party eligible for auto delegation: (0059-STKG-020)
- epoch 0: party associated 1000 VEGA
- epoch 0: party nominated 200 VEGA to validators 1-5
- epoch 1: party associated 200 VEGA
- epoch 1: party requests to delegate 100 VEGA to
validator1
- end of epoch1: party1 has 300 delegated to validator1, 200 delegated to validators 2-5 and 100 remain undelegated.
- end of epoch2: the remaining associated undelegated 100 VEGA get auto-delegated and distributed such that
validator1
gets 27 (100 * 300/1100) and validators 2-5 get each 18 - and 1 token remains undelegated
Edge case 2: manual undelegation for party eligible for auto delegation: (0059-STKG-021)
- epoch 0: party associated 1000 VEGA
- epoch 0: party nominated 200 VEGA to validators 1-5
- epoch 1: party associated 100 VEGA
- epoch 1: party requests to undelegate 200 VEGA from
validator1
- end of epoch1: party has 300 unnominated VEGA which will NOT be auto delegated
- epoch 2: party requests to delegate 300 to
validator2
- epoch 2: party associated 100 VEGA
- end of epoch 2: party has 500 nominated to
validator2
and 200 nominated to validators 3-5 - end of epoch 3: party has 100 unnominated VEGA which gets nominated proportionally between validators 2-5 - i.e.
validator2
gets 45, validators 3-4-5 get 18 each
- 0013-ACCT Acccounts - staking accounts are not like other account types, but the differences are covered here.
- 0028-GOVE Governance - a party's stake controls their ability to participate in governance.
- 0069-VALW Validators chosen by stake - staking and delegation is used to pick validators.
- 0050-EPOC Epochs - epochs control how frequently validator sets can change as a result of staking and delegation.