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

Feat/withdrawal credentials #904

Open
wants to merge 12 commits into
base: develop
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 4 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
49 changes: 39 additions & 10 deletions contracts/0.8.9/WithdrawalVault.sol
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import "@openzeppelin/contracts-v4.4/token/ERC721/IERC721.sol";
import "@openzeppelin/contracts-v4.4/token/ERC20/utils/SafeERC20.sol";

import {Versioned} from "./utils/Versioned.sol";
import {TriggerableWithdrawals} from "./lib/TriggerableWithdrawals.sol";

interface ILido {
/**
Expand All @@ -27,6 +28,7 @@ contract WithdrawalVault is Versioned {

ILido public immutable LIDO;
address public immutable TREASURY;
address public immutable VALIDATORS_EXIT_BUS;
mkurayan marked this conversation as resolved.
Show resolved Hide resolved

// Events
/**
Expand All @@ -42,26 +44,24 @@ contract WithdrawalVault is Versioned {
event ERC721Recovered(address indexed requestedBy, address indexed token, uint256 tokenId);

// Errors
error LidoZeroAddress();
error TreasuryZeroAddress();
error ZeroAddress();
error NotLido();
error NotValidatorExitBus();
error NotEnoughEther(uint256 requested, uint256 balance);
error ZeroAmount();

/**
* @param _lido the Lido token (stETH) address
* @param _treasury the Lido treasury address (see ERC20/ERC721-recovery interfaces)
*/
constructor(ILido _lido, address _treasury) {
if (address(_lido) == address(0)) {
revert LidoZeroAddress();
}
if (_treasury == address(0)) {
revert TreasuryZeroAddress();
}
constructor(address _lido, address _treasury, address _validatorsExitBus) {
_requireNonZero(_lido);
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
_requireNonZero(_lido);
_onlyNonZeroAddress(_lido);

_requireNonZero(_treasury);
_requireNonZero(_validatorsExitBus);

LIDO = _lido;
LIDO = ILido(_lido);
TREASURY = _treasury;
VALIDATORS_EXIT_BUS = _validatorsExitBus;
}

/**
Expand All @@ -70,6 +70,12 @@ contract WithdrawalVault is Versioned {
*/
function initialize() external {
_initializeContractVersionTo(1);
_updateContractVersion(2);
}

function finalizeUpgrade_v2() external {
_checkContractVersion(1);
_updateContractVersion(2);
}

/**
Expand Down Expand Up @@ -122,4 +128,27 @@ contract WithdrawalVault is Versioned {

_token.transferFrom(address(this), TREASURY, _tokenId);
}

/**
* @dev Adds full withdrawal requests for the provided public keys.
* The validator will fully withdraw and exit its duties as a validator.
* @param pubkeys An array of public keys for the validators requesting full withdrawals.
*/
function addFullWithdrawalRequests(
bytes[] calldata pubkeys
) external payable {
if(msg.sender != address(VALIDATORS_EXIT_BUS)) {
revert NotValidatorExitBus();
}

TriggerableWithdrawals.addFullWithdrawalRequests(pubkeys, msg.value);
}

function getWithdrawalRequestFee() external view returns (uint256) {
return TriggerableWithdrawals.getWithdrawalRequestFee();
}

function _requireNonZero(address _address) internal pure {
if (_address == address(0)) revert ZeroAddress();
}
}
148 changes: 148 additions & 0 deletions contracts/0.8.9/lib/TriggerableWithdrawals.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,148 @@
// SPDX-FileCopyrightText: 2023 Lido <[email protected]>
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
// SPDX-FileCopyrightText: 2023 Lido <[email protected]>
// SPDX-FileCopyrightText: 2025 Lido <[email protected]>

// SPDX-License-Identifier: GPL-3.0

pragma solidity 0.8.9;
Copy link
Member

Choose a reason for hiding this comment

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

Suggested change
pragma solidity 0.8.9;
// solhint-disable-next-line lido/fixed-compiler-version
pragma solidity >=0.8.9 <0.9.0;

Also, consider moving to common libs?


library TriggerableWithdrawals {
address constant WITHDRAWAL_REQUEST = 0x0c15F14308530b7CDB8460094BbB9cC28b9AaaAA;
Copy link
Member

Choose a reason for hiding this comment

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

Add some comment with "validation" link?

Suggested change
address constant WITHDRAWAL_REQUEST = 0x0c15F14308530b7CDB8460094BbB9cC28b9AaaAA;
/// @dev https://eips.ethereum.org/EIPS/eip-7002#configuration
address constant WITHDRAWAL_REQUEST = 0x0c15F14308530b7CDB8460094BbB9cC28b9AaaAA;

Copy link
Member

@tamtamchik tamtamchik Jan 20, 2025

Choose a reason for hiding this comment

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

May this contract address be changed on testnets to something else (as, for example, with the deposit contract on Holesky)? We'll have to use a separate library for it?


error MismatchedArrayLengths(uint256 keysCount, uint256 amountsCount);
error InsufficientBalance(uint256 balance, uint256 totalWithdrawalFee);
error FeeNotEnough(uint256 minFeePerRequest, uint256 requestCount, uint256 providedTotalFee);

error WithdrawalRequestFeeReadFailed();
error InvalidPubkeyLength(bytes pubkey);
error WithdrawalRequestAdditionFailed(bytes pubkey, uint256 amount);
error NoWithdrawalRequests();
error PartialWithdrawalRequired(bytes pubkey);

event WithdrawalRequestAdded(bytes pubkey, uint256 amount);

/**
* @dev Adds full withdrawal requests for the provided public keys.
* The validator will fully withdraw and exit its duties as a validator.
* @param pubkeys An array of public keys for the validators requesting full withdrawals.
*/
function addFullWithdrawalRequests(
bytes[] calldata pubkeys,
uint256 totalWithdrawalFee
Copy link
Member

Choose a reason for hiding this comment

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

It think that this parameter is not required at all. We can safely assume that all required ether is on the balance of the contract and we'll revert if it's not true and if we need some additional constraints (like, msg.value == fee), we can add it in the contract that use that lib.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

This parameter was introduced to decouple the fee allocation strategy from the withdrawals library, discussion. It enables contracts to employ different allocation strategies.

The proposed Validator Exitt Bus Triggerable Withdrawal implementation assumes that the withdrawal fee is provided by the actor who triggers the withdrawals. In this approach, the WithdrawalVault.sol balance remains unaffected, preventing issues with the Oracle’s accounting. Consequently, the entire msg.value sent is used as the fee for withdrawal requests:

// Simplified pseudo-code
function addWithdrawalRequests(
    bytes[] calldata pubkeys,
    uint64[] calldata amounts
) external payable {
    // Use the entire sent amount (msg.value) as the total fee for withdrawal requests
    uint256 totalWithdrawalFee = msg.value;
    WithdrawalRequests.addWithdrawalRequests(pubkeys, amounts, totalWithdrawalFee);
}

Other vaults could employ the strategy you mentioned, assuming all required Ether is already in the contract’s balance:

// Simplified pseudo-code
function addWithdrawalRequests(
    bytes[] calldata pubkeys,
    uint64[] calldata amounts
) external {
    // Use the minimum required fee per request
    uint256 minFeePerRequest = WithdrawalRequests.getWithdrawalRequestFee();
    uint256 totalWithdrawalFee = minFeePerRequest * pubkeys.length;
    WithdrawalRequests.addWithdrawalRequests(pubkeys, amounts, totalWithdrawalFee);
}

When the withdrawal fee is specified explicitly, any fee allocation strategy can be used. The library ensures that the provided fee sufficiently covers all requests and that the exact fee amount is spent.

Copy link
Member

Choose a reason for hiding this comment

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

When the withdrawal fee is specified explicitly, any fee allocation strategy can be used. The library ensures that the provided fee sufficiently covers all requests and that the exact fee amount is spent.

See no reason to pass it inside the function when it can definitely be checked before and after the function call in the WithdrawalVault itself. It's kinda alien constraint for the raw withdrawal request creation library.

E.g. in the vaults we don't care where the funds for the gas will come from and we don't need to check it at all.

Copy link
Contributor Author

@mkurayan mkurayan Jan 14, 2025

Choose a reason for hiding this comment

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

The EIP-7002 specification does not impose an upper limit on the withdrawal request fee; this was intentionally designed for flexibility. The library implementation proposed in this PR follows the EIP-7002 specification and does not add any extra restrictions on withdrawal requests.

If we do not want the general purpose library, we can remove control over the request fee, and narrow the library's functionality to allow only requests with minimal fees. This would simplify the code slightly, but also limit the library's potential use cases.

I am not confident that we will not need control over the request fee in the future, @folkyatina what do you think?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

After an internal discussion, we agreed to follow the eip 7002 specification and keep control over the request fee amount, but pass the request fee instead of the total withdrawal fee to simplify library implementation.

) internal {
uint64[] memory amounts = new uint64[](pubkeys.length);
_addWithdrawalRequests(pubkeys, amounts, totalWithdrawalFee);
}

/**
* @dev Adds partial withdrawal requests for the provided public keys with corresponding amounts.
* A partial withdrawal is any withdrawal where the amount is greater than zero.
* A full withdrawal is any withdrawal where the amount is zero.
* This allows withdrawal of any balance exceeding 32 ETH (e.g., if a validator has 35 ETH, up to 3 ETH can be withdrawn).
* However, the protocol enforces a minimum balance of 32 ETH per validator, even if a higher amount is requested.
* @param pubkeys An array of public keys for the validators requesting withdrawals.
* @param amounts An array of corresponding withdrawal amounts for each public key.
*/
function addPartialWithdrawalRequests(
bytes[] calldata pubkeys,
uint64[] calldata amounts,
uint256 totalWithdrawalFee
) internal {
_requireArrayLengthsMatch(pubkeys, amounts);

for (uint256 i = 0; i < amounts.length; i++) {
if (amounts[i] == 0) {
revert PartialWithdrawalRequired(pubkeys[i]);
}
}

_addWithdrawalRequests(pubkeys, amounts, totalWithdrawalFee);
}

/**
* @dev Adds partial or full withdrawal requests for the provided public keys with corresponding amounts.
* A partial withdrawal is any withdrawal where the amount is greater than zero.
* This allows withdrawal of any balance exceeding 32 ETH (e.g., if a validator has 35 ETH, up to 3 ETH can be withdrawn).
* However, the protocol enforces a minimum balance of 32 ETH per validator, even if a higher amount is requested.
* @param pubkeys An array of public keys for the validators requesting withdrawals.
* @param amounts An array of corresponding withdrawal amounts for each public key.
*/
function addWithdrawalRequests(
bytes[] calldata pubkeys,
uint64[] calldata amounts,
uint256 totalWithdrawalFee
) internal {
_requireArrayLengthsMatch(pubkeys, amounts);
_addWithdrawalRequests(pubkeys, amounts, totalWithdrawalFee);
}

/**
* @dev Retrieves the current withdrawal request fee.
* @return The minimum fee required per withdrawal request.
*/
function getWithdrawalRequestFee() internal view returns (uint256) {
(bool success, bytes memory feeData) = WITHDRAWAL_REQUEST.staticcall("");

if (!success) {
revert WithdrawalRequestFeeReadFailed();
}

return abi.decode(feeData, (uint256));
}

function _addWithdrawalRequests(
bytes[] calldata pubkeys,
uint64[] memory amounts,
uint256 totalWithdrawalFee
) internal {
uint256 keysCount = pubkeys.length;
if (keysCount == 0) {
revert NoWithdrawalRequests();
}

if(address(this).balance < totalWithdrawalFee) {
revert InsufficientBalance(address(this).balance, totalWithdrawalFee);
}

uint256 minFeePerRequest = getWithdrawalRequestFee();
folkyatina marked this conversation as resolved.
Show resolved Hide resolved
if (minFeePerRequest * keysCount > totalWithdrawalFee) {
revert FeeNotEnough(minFeePerRequest, keysCount, totalWithdrawalFee);
}

uint256 feePerRequest = totalWithdrawalFee / keysCount;
uint256 unallocatedFee = totalWithdrawalFee % keysCount;
uint256 prevBalance = address(this).balance - totalWithdrawalFee;

for (uint256 i = 0; i < keysCount; ++i) {
bytes memory pubkey = pubkeys[i];
folkyatina marked this conversation as resolved.
Show resolved Hide resolved
uint64 amount = amounts[i];

if(pubkey.length != 48) {
revert InvalidPubkeyLength(pubkey);
}

uint256 feeToSend = feePerRequest;

if (i == keysCount - 1) {
feeToSend += unallocatedFee;
}

bytes memory callData = abi.encodePacked(pubkey, amount);
(bool success, ) = WITHDRAWAL_REQUEST.call{value: feeToSend}(callData);

if (!success) {
revert WithdrawalRequestAdditionFailed(pubkey, amount);
}

emit WithdrawalRequestAdded(pubkey, amount);
}

assert(address(this).balance == prevBalance);
}
Fixed Show fixed Hide fixed

function _requireArrayLengthsMatch(
bytes[] calldata pubkeys,
uint64[] calldata amounts
) internal pure {
if (pubkeys.length != amounts.length) {
revert MismatchedArrayLengths(pubkeys.length, amounts.length);
}
}
}
38 changes: 38 additions & 0 deletions test/0.8.9/contracts/TriggerableWithdrawals_Harness.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
pragma solidity 0.8.9;

import {TriggerableWithdrawals} from "contracts/0.8.9/lib/TriggerableWithdrawals.sol";

contract TriggerableWithdrawals_Harness {
function addFullWithdrawalRequests(
bytes[] calldata pubkeys,
uint256 totalWithdrawalFee
) external {
TriggerableWithdrawals.addFullWithdrawalRequests(pubkeys, totalWithdrawalFee);
}

function addPartialWithdrawalRequests(
bytes[] calldata pubkeys,
uint64[] calldata amounts,
uint256 totalWithdrawalFee
) external {
TriggerableWithdrawals.addPartialWithdrawalRequests(pubkeys, amounts, totalWithdrawalFee);
}

function addWithdrawalRequests(
bytes[] calldata pubkeys,
uint64[] calldata amounts,
uint256 totalWithdrawalFee
) external {
TriggerableWithdrawals.addWithdrawalRequests(pubkeys, amounts, totalWithdrawalFee);
}

function getWithdrawalRequestFee() external view returns (uint256) {
return TriggerableWithdrawals.getWithdrawalRequestFee();
}

function getWithdrawalsContractAddress() public pure returns (address) {
return TriggerableWithdrawals.WITHDRAWAL_REQUEST;
}

function deposit() external payable {}
}
37 changes: 37 additions & 0 deletions test/0.8.9/contracts/predeployed/WithdrawalsPredeployed_Mock.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
// SPDX-License-Identifier: UNLICENSED
pragma solidity 0.8.9;

/**
* @notice This is an mock of EIP-7002's pre-deploy contract.
*/
contract WithdrawalsPredeployed_Mock {
uint256 public fee;
bool public failOnAddRequest;
bool public failOnGetFee;

function setFailOnAddRequest(bool _failOnAddRequest) external {
failOnAddRequest = _failOnAddRequest;
}

function setFailOnGetFee(bool _failOnGetFee) external {
failOnGetFee = _failOnGetFee;
}

function setFee(uint256 _fee) external {
require(_fee > 0, "fee must be greater than 0");
fee = _fee;
}

fallback(bytes calldata input) external payable returns (bytes memory output){
if (input.length == 0) {
require(!failOnGetFee, "fail on get fee");

output = abi.encode(fee);
return output;
}

require(!failOnAddRequest, "fail on add request");

require(input.length == 56, "Invalid callData length");
}
}
13 changes: 13 additions & 0 deletions test/0.8.9/lib/triggerableWithdrawals/findEvents.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import { ContractTransactionReceipt } from "ethers";
import { ethers } from "hardhat";

import { findEventsWithInterfaces } from "lib";

const withdrawalRequestEventABI = ["event WithdrawalRequestAdded(bytes pubkey, uint256 amount)"];
const withdrawalRequestEventInterface = new ethers.Interface(withdrawalRequestEventABI);

type WithdrawalRequestEvents = "WithdrawalRequestAdded";

export function findEvents(receipt: ContractTransactionReceipt, event: WithdrawalRequestEvents) {
return findEventsWithInterfaces(receipt!, event, [withdrawalRequestEventInterface]);
}
Loading
Loading