cover | coverY | layout | ||||||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
../.gitbook/assets/Frame 27513376 (1).png |
0 |
|
Oracles updating rewards
Currently, there is no way to fetch the Beacon Chain state in the EVM. It is planned to make it available in EIP-4788. Until then, StakeWise uses Oracles to fetch rewards from the Beacon Chain and submit the updates to the contracts. The Oracles are listed here. The Oracles are selected and approved by the StakeWise DAO. The Oracles run the v3-oracle nodes and have the following responsibilities:
The Oracles periodically vote for the rewards/penalties accumulated by the Vaults in the Beacon Chain and execution rewards (MEV & priority fees) for the Vaults connected to the smoothing pool.
The voting process consists of the following steps:
- Check whether 12 hours passed since the last rewards distribution
- Calculate consensus rewards/penalties for all the Vaults based on their validator balances in the Beacon Chain.
- Calculate execution rewards/penalties for the Vaults connected to the smoothing pool.
- Calculate the Merkle tree based on consensus/execution rewards from the previous steps and upload it to IPFS. For example, bafkreibqhdr6p5uh67ickt4dpppb525bwuofjocnpsx4dbl57llogfph2e.
- Save the vote to the database and make it available through public API.
Anyone who runs v3-keeper will fetch votes from Oracles endpoints, concatenate them and send them to the Keeper contract. Currently, 6 out of 11 votes are required to submit an update. Once the update has submitted, Vaults can pull the updates through the updateState
call.
The operators (v3-operator) periodically check whether their Vaults have accumulated enough ETH for registering new validator(s) and send a registration approval request to oracles. This is done to receive an exit signature for the validator so that it could be later exited in case the staker submits an exit request and to protect from the attack described here. The operator must receive 8 out of 11 approvals from the oracles to register a validator for the Vault.
The validator registration process consists of the following steps:
- The operator sends a registration request with a payload to the oracles.
- Oracles validate operator request.
- Oracles sign and return the registration approval to the operator if it is correct.
- The operator submits the registration transaction to the Vault contract.
- The Vault contract validates oracles' signatures and moves ETH from the Vault to the Beacon Chain contract.
Once oracles submit a new Merkle tree root to the Keeper contract, every Vault can pull these updates by calling vault.updateState
. To check whether updates can be pulled from the Keeper contract, the keeper.canHarvest
can be called. The state becomes outdated once the Vault hasn't pulled two consequent updates. To check whether the state is outdated, call a keeper.isHarvestRequired
.
The Vault state can be updated in one of the following ways:
- By the user during interaction with the Vault. This can be done by wrapping the call (e.g., redeem, osETH mint, etc.) with a
multicall
and adding a state update. For example,
const balance = await vaultContract.balanceOf(staker.address)
let tx
if (await keeperContract.canHarvest(vaultContract.address)) {
const calls = [
vaultContract.interface.encodeFunctionData('updateState', [harvestParams]),
vaultContract.interface.encodeFunctionData('redeem', [balance, staker.address]),
]
tx = await vaultContract.connect(staker).multicall(calls)
} else {
tx = await vaultContract.connect(staker).redeem(balance, staker.address)
}
{% hint style="info" %}
The harvestParams can be fetched from the StakeWise subgraph or by fetching the last RewardsUpdated event from the Keeper contract, extracting the rewardsIpfsHash
, fetching the file, and getting the parameters for your Vault from it.
{% endhint %}
- By the Vault operator by passing
--harvest-vault
to thestart
command in v3-operator.