pragma solidity ^0.8.13;
import "src/util/OffchainAbstractHelper.sol";
interface ICurvePool {
function coins(uint index) external view returns(address);
function get_dy(uint i, uint j, uint dx) external view returns(uint);
function exchange(uint i, uint j, uint dx, uint min_dy, bool use_eth) external payable returns(uint);
function exchange(uint i, uint j, uint dx, uint min_dy, bool use_eth, address receiver) external payable returns(uint);
}
contract CurveHelper is OffchainAbstractHelper{
ICurvePool public immutable curvePool;
uint dbrIndex;
uint dolaIndex;
constructor(address _pool, uint tokens) {
curvePool = ICurvePool(_pool);
DOLA.approve(_pool, type(uint).max);
DBR.approve(_pool, type(uint).max);
for(uint i; i < tokens; ++i){
if(ICurvePool(_pool).coins(i) == address(DOLA)){
dolaIndex = i;
}
else if(ICurvePool(_pool).coins(i) == address(DBR)){
dbrIndex = i;
}
}
}
/**
@notice Sells an exact amount of DBR for DOLA in a curve pool
@param amount Amount of DBR to sell
@param minOut minimum amount of DOLA to receive
*/
function _sellDbr(uint amount, uint minOut) internal override {
if(amount > 0){
curvePool.exchange(dbrIndex, dolaIndex, amount, minOut, false);
}
}
/**
@notice Buys an exact amount of DBR for DOLA in a curve pool
@param amount Amount of DOLA to sell
@param minOut minimum amount of DBR out
*/
function _buyDbr(uint amount, uint minOut, address receiver) internal override {
if(amount > 0) {
curvePool.exchange(dolaIndex, dbrIndex, amount, minOut, false, receiver);
}
}
/**
@notice Approximates the total amount of dola and dbr needed to borrow a dolaBorrowAmount while also borrowing enough to buy the DBR needed to cover for the borrowing period
@dev Uses a binary search to approximate the amounts needed. Should only be called as part of generating transaction parameters.
@param dolaBorrowAmount Amount of dola the user wishes to end up with
@param period Amount of time in seconds the loan will last
@param iterations Number of approximation iterations. The higher the more precise the result
*/
function approximateDolaAndDbrNeeded(uint dolaBorrowAmount, uint period, uint iterations) public view override returns(uint dolaForDbr, uint dbrNeeded){
uint amountIn = dolaBorrowAmount;
uint stepSize = amountIn / 2;
uint dbrReceived = curvePool.get_dy(dolaIndex, dbrIndex, amountIn);
uint dbrToBuy = (amountIn + dolaBorrowAmount) * period / 365 days;
uint dist = dbrReceived > dbrToBuy ? dbrReceived - dbrToBuy : dbrToBuy - dbrReceived;
for(uint i; i < iterations; ++i){
uint newAmountIn = amountIn;
if(dbrReceived > dbrToBuy){
newAmountIn -= stepSize;
} else {
newAmountIn += stepSize;
}
uint newDbrReceived = curvePool.get_dy(dolaIndex, dbrIndex, newAmountIn);
uint newDbrToBuy = (newAmountIn + dolaBorrowAmount) * period / 365 days;
uint newDist = newDbrReceived > newDbrToBuy ? newDbrReceived - newDbrToBuy : newDbrToBuy - newDbrReceived;
if(newDist < dist){
dbrReceived = newDbrReceived;
dbrToBuy = newDbrToBuy;
dist = newDist;
amountIn = newAmountIn;
}
stepSize /= 2;
}
return (amountIn, (dolaBorrowAmount + amountIn) * period / 365 days);
}
}
pragma solidity ^0.8.13;
interface IMarket {
function borrowOnBehalf(address msgSender, uint dolaAmount, uint deadline, uint8 v, bytes32 r, bytes32 s) external;
function withdrawOnBehalf(address msgSender, uint amount, uint deadline, uint8 v, bytes32 r, bytes32 s) external;
function deposit(address msgSender, uint collateralAmount) external;
function repay(address msgSender, uint amount) external;
function collateral() external returns(address);
function debts(address user) external returns(uint);
function recall(uint amount) external;
function totalDebt() external view returns (uint);
function borrowPaused() external view returns (bool);
}
pragma solidity ^0.8.13;
import "../interfaces/IMarket.sol";
interface IERC20 {
function transfer(address to, uint amount) external;
function transferFrom(address from, address to, uint amount) external;
function approve(address to, uint amount) external;
function balanceOf(address user) external view returns(uint);
}
interface IWETH is IERC20 {
function withdraw(uint wad) external;
function deposit() external payable;
}
abstract contract OffchainAbstractHelper {
IERC20 constant DOLA = IERC20(0x865377367054516e17014CcdED1e7d814EDC9ce4);
IERC20 constant DBR = IERC20(0xAD038Eb671c44b853887A7E32528FaB35dC5D710);
IWETH constant WETH = IWETH(0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2);
/**
Virtual functions implemented by the AMM interfacing part of the Helper contract
*/
/**
@notice Buys DBR for an amount of Dola
@param amount Amount of Dola to spend on DBR
@param minOut minimum amount of DBR to receive
*/
function _buyDbr(uint amount, uint minOut, address receiver) internal virtual;
/**
@notice Sells an exact amount of DBR for DOLA
@param amount Amount of DBR to sell
@param minOut minimum amount of DOLA to receive
*/
function _sellDbr(uint amount, uint minOut) internal virtual;
/**
@notice Approximates the amount of additional DOLA and DBR needed to sustain dolaBorrowAmount over the period
@dev Larger number of iterations increases both accuracy of the approximation and gas cost. This method should not be called in smart contract code.
@param dolaBorrowAmount The amount of DOLA the user wishes to borrow before covering DBR expenses
@param minDbr The amount of seconds the user wish to borrow the DOLA for
@param iterations The amount of approximation iterations.
@return Tuple of (dolaNeeded, dbrNeeded) representing the total dola needed to pay for the DBR and pay out dolaBorrowAmount and the dbrNeeded to sustain the loan over the period
*/
function approximateDolaAndDbrNeeded(uint dolaBorrowAmount, uint minDbr, uint iterations) public view virtual returns(uint, uint);
/**
@notice Borrows on behalf of the caller, buying the necessary DBR to pay for the loan over the period, by borrowing aditional funds to pay for the necessary DBR
@dev Has to borrow the dolaForDbr amount due to how the market's borrowOnBehalf functions, and repay the excess at the end of the call resulting in a weird repay event
@param market Market the caller wishes to borrow from
@param dolaAmount Amount the caller wants to end up with at their disposal
@param dolaForDbr The max amount of debt the caller is willing to end up with
This is a sensitive parameter and should be reasonably low to prevent sandwhiching.
A good estimate can be calculated given the approximateDolaAndDbrNeeded function, though should be set slightly higher.
@param minDbr The minDbr the caller wish to borrow for
@param deadline Deadline of the signature
@param v V parameter of the signature
@param r R parameter of the signature
@param s S parameter of the signature
*/
function buyDbrAndBorrowOnBehalf(
IMarket market,
uint dolaAmount,
uint dolaForDbr,
uint minDbr,
uint deadline,
uint8 v,
bytes32 r,
bytes32 s)
public
{
//Borrow Dola
uint totalBorrow = dolaAmount + dolaForDbr;
market.borrowOnBehalf(msg.sender, totalBorrow, deadline, v, r, s);
//Buy DBR
_buyDbr(dolaForDbr, minDbr, msg.sender);
//Transfer remaining DOLA amount to user
DOLA.transfer(msg.sender, dolaAmount);
}
/**
@notice Deposits collateral and borrows on behalf of the caller, buying the necessary DBR to pay for the loan over the period, by borrowing aditional funds to pay for the necessary DBR
@dev Has to borrow the dolaForDbr amount due to how the market's borrowOnBehalf functions, and repay the excess at the end of the call resulting in a weird repay event
@param market Market the caller wish to deposit to and borrow from
@param dolaAmount Amount the caller wants to end up with at their disposal
@param dolaForDbr The max amount of debt the caller is willing to take on to buy dbr
This is a sensitive parameter and should be reasonably low to prevent sandwhiching.
A good estimate can be calculated given the approximateDolaAndDbrNeeded function, though should be set slightly higher.
@param minDbr The minDbr the caller wish to borrow for
@param deadline Deadline of the signature
@param v V parameter of the signature
@param r R parameter of the signature
@param s S parameter of the signature
*/
function depositBuyDbrAndBorrowOnBehalf(
IMarket market,
uint collateralAmount,
uint dolaAmount,
uint dolaForDbr,
uint minDbr,
uint deadline,
uint8 v,
bytes32 r,
bytes32 s)
public
{
IERC20 collateral = IERC20(market.collateral());
//Deposit collateral
collateral.transferFrom(msg.sender, address(this), collateralAmount);
collateral.approve(address(market), collateralAmount);
market.deposit(msg.sender, collateralAmount);
//Borrow dola and buy dbr
buyDbrAndBorrowOnBehalf(market, dolaAmount, dolaForDbr, minDbr, deadline, v, r , s);
}
/**
@notice Deposits native eth as collateral and borrows on behalf of the caller,
buying the necessary DBR to pay for the loan over the period, by borrowing aditional funds to pay for the necessary DBR
@dev Has to borrow the dolaForDbr amount due to how the market's borrowOnBehalf functions, and repay the excess at the end of the call resulting in a weird repay event
@param market Market the caller wish to deposit to and borrow from
@param dolaAmount Amount the caller wants to end up with at their disposal
@param dolaForDbr The max amount of debt the caller is willing to end up with
This is a sensitive parameter and should be reasonably low to prevent sandwhiching.
A good estimate can be calculated given the approximateDolaAndDbrNeeded function, though should be set slightly higher.
@param minDbr The minDbr the caller wish to borrow for
@param deadline Deadline of the signature
@param v V parameter of the signature
@param r R parameter of the signature
@param s S parameter of the signature
*/
function depositNativeEthBuyDbrAndBorrowOnBehalf(
IMarket market,
uint dolaAmount,
uint dolaForDbr,
uint minDbr,
uint deadline,
uint8 v,
bytes32 r,
bytes32 s)
public payable
{
IERC20 collateral = IERC20(market.collateral());
require(address(collateral) == address(WETH), "Market is not an ETH market");
WETH.deposit{value:msg.value}();
//Deposit collateral
collateral.approve(address(market), msg.value);
market.deposit(msg.sender, msg.value);
//Borrow dola and buy dbr
buyDbrAndBorrowOnBehalf(market, dolaAmount, dolaForDbr, minDbr, deadline, v, r , s);
}
/**
@notice Sells DBR on behalf of the caller and uses the proceeds along with DOLA from the caller to repay debt.
@dev The caller is unlikely to spend all of the DOLA they make available for the function call
@param market The market the user wishes to repay debt in
@param dolaAmount The maximum amount of dola debt the user is willing to repay
@param minDolaFromDbr The minimum amount of DOLA the caller expects to get in return for selling their DBR.
This is a sensitive parameter and should be provided with reasonably low slippage to prevent sandwhiching.
@param dbrAmountToSell The amount of DBR the caller wishes to sell
*/
function sellDbrAndRepayOnBehalf(IMarket market, uint dolaAmount, uint minDolaFromDbr, uint dbrAmountToSell) public {
uint dbrBal = DBR.balanceOf(msg.sender);
//If user has less DBR than ordered, sell what's available
if(dbrAmountToSell > dbrBal){
DBR.transferFrom(msg.sender, address(this), dbrBal);
_sellDbr(dbrBal, minDolaFromDbr);
} else {
DBR.transferFrom(msg.sender, address(this), dbrAmountToSell);
_sellDbr(dbrAmountToSell, minDolaFromDbr);
}
uint debt = market.debts(msg.sender);
uint dolaBal = DOLA.balanceOf(address(this));
//If the debt is lower than the dolaAmount, repay debt else repay dolaAmount
uint repayAmount = debt < dolaAmount ? debt : dolaAmount;
//If dolaBal is less than repayAmount, transfer remaining DOLA from user, otherwise transfer excess dola to user
if(dolaBal < repayAmount){
DOLA.transferFrom(msg.sender, address(this), repayAmount - dolaBal);
} else {
DOLA.transfer(msg.sender, dolaBal - repayAmount);
}
//Repay repayAmount
DOLA.approve(address(market), repayAmount);
market.repay(msg.sender, repayAmount);
}
/**
@notice Sells DBR on behalf of the caller and uses the proceeds along with DOLA from the caller to repay debt, and then withdraws collateral
@dev The caller is unlikely to spend all of the DOLA they make available for the function call
@param market Market the user wishes to repay debt in
@param dolaAmount Maximum amount of dola debt the user is willing to repay
@param minDolaFromDbr Minimum amount of DOLA the caller expects to get in return for selling their DBR
This is a sensitive parameter and should be provided with reasonably low slippage to prevent sandwhiching.
@param dbrAmountToSell Amount of DBR the caller wishes to sell
@param collateralAmount Amount of collateral to withdraw
@param deadline Deadline of the signature
@param v V parameter of the signature
@param r R parameter of the signature
@param s S parameter of the signature
*/
function sellDbrRepayAndWithdrawOnBehalf(
IMarket market,
uint dolaAmount,
uint minDolaFromDbr,
uint dbrAmountToSell,
uint collateralAmount,
uint deadline,
uint8 v,
bytes32 r,
bytes32 s)
external
{
//Repay
sellDbrAndRepayOnBehalf(market, dolaAmount, minDolaFromDbr, dbrAmountToSell);
//Withdraw
market.withdrawOnBehalf(msg.sender, collateralAmount, deadline, v, r, s);
//Transfer collateral to msg.sender
IERC20(market.collateral()).transfer(msg.sender, collateralAmount);
}
/**
@notice Sells DBR on behalf of the caller and uses the proceeds along with DOLA from the caller to repay debt, and then withdraws collateral
@dev The caller is unlikely to spend all of the DOLA they make available for the function call
@param market Market the user wishes to repay debt in
@param dolaAmount Maximum amount of dola debt the user is willing to repay
@param minDolaFromDbr Minimum amount of DOLA the caller expects to get in return for selling their DBR
This is a sensitive parameter and should be provided with reasonably low slippage to prevent sandwhiching.
@param dbrAmountToSell Amount of DBR the caller wishes to sell
@param collateralAmount Amount of collateral to withdraw
@param deadline Deadline of the signature
@param v V parameter of the signature
@param r R parameter of the signature
@param s S parameter of the signature
*/
function sellDbrRepayAndWithdrawNativeEthOnBehalf(
IMarket market,
uint dolaAmount,
uint minDolaFromDbr,
uint dbrAmountToSell,
uint collateralAmount,
uint deadline,
uint8 v,
bytes32 r,
bytes32 s)
external
{
//Repay
sellDbrAndRepayOnBehalf(market, dolaAmount, minDolaFromDbr, dbrAmountToSell);
//Withdraw
withdrawNativeEthOnBehalf(market, collateralAmount, deadline, v, r, s);
}
/**
@notice Repays debt, and then withdraws native ETH
@dev The caller is unlikely to spend all of the DOLA they make available for the function call
@param market Market the user wishes to repay debt in
@param dolaAmount Amount of dola debt the user is willing to repay
@param collateralAmount Amount of collateral to withdraw
@param deadline Deadline of the signature
@param v V parameter of the signature
@param r R parameter of the signature
@param s S parameter of the signature
*/
function repayAndWithdrawNativeEthOnBehalf(
IMarket market,
uint dolaAmount,
uint collateralAmount,
uint deadline,
uint8 v,
bytes32 r,
bytes32 s)
external
{
// Repay
DOLA.transferFrom(msg.sender, address(this), dolaAmount);
DOLA.approve(address(market), dolaAmount);
market.repay(msg.sender, dolaAmount);
// Withdraw
withdrawNativeEthOnBehalf(market, collateralAmount, deadline, v, r, s);
}
/**
@notice Helper function for depositing native eth to WETH markets
@param market The WETH market to deposit to
*/
function depositNativeEthOnBehalf(IMarket market) public payable {
require(address(market.collateral()) == address(WETH), "Not an ETH market");
WETH.deposit{value:msg.value}();
WETH.approve(address(market), msg.value);
market.deposit(msg.sender, msg.value);
}
/**
@notice Helper function for depositing native eth to WETH markets before borrowing on behalf of the depositor
@param market The WETH market to deposit to
@param borrowAmount The amount to borrow on behalf of the depositor
@param deadline Deadline of the signature
@param v V parameter of the signature
@param r R parameter of the signature
@param s S parameter of the signature
*/
function depositNativeEthAndBorrowOnBehalf(IMarket market, uint borrowAmount, uint deadline, uint8 v, bytes32 r, bytes32 s) public payable {
require(address(market.collateral()) == address(WETH), "Not an ETH market");
//Deposit native eth
WETH.deposit{value:msg.value}();
WETH.approve(address(market), msg.value);
market.deposit(msg.sender, msg.value);
//Borrow Dola
market.borrowOnBehalf(msg.sender, borrowAmount, deadline, v, r, s);
DOLA.transfer(msg.sender, borrowAmount);
}
/**
@notice Helper function for withdrawing to native eth
@param market WETH market to withdraw collateral from
@param collateralAmount Amount of collateral to withdraw
@param deadline Deadline of the signature
@param v V parameter of the signature
@param r R parameter of the signature
@param s S parameter of the signature
*/
function withdrawNativeEthOnBehalf(
IMarket market,
uint collateralAmount,
uint deadline,
uint8 v,
bytes32 r,
bytes32 s)
public
{
market.withdrawOnBehalf(msg.sender, collateralAmount, deadline, v, r, s);
IERC20 collateral = IERC20(market.collateral());
require(address(collateral) == address(WETH), "Not an ETH market");
WETH.withdraw(collateralAmount);
(bool success,) = payable(msg.sender).call{value:collateralAmount}("");
require(success, "Failed to transfer ETH");
}
//Empty receive function for receiving the native eth sent by the WETH contract
receive() external payable {}
}
{
"compilationTarget": {
"src/util/CurveHelper.sol": "CurveHelper"
},
"evmVersion": "shanghai",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": [
":ds-test/=lib/solmate/lib/ds-test/src/",
":forge-std/=lib/forge-std/src/",
":openzeppelin-contracts/=lib/openzeppelin-contracts/",
":solmate/=lib/solmate/src/"
]
}
[{"inputs":[{"internalType":"address","name":"_pool","type":"address"},{"internalType":"uint256","name":"tokens","type":"uint256"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[{"internalType":"uint256","name":"dolaBorrowAmount","type":"uint256"},{"internalType":"uint256","name":"period","type":"uint256"},{"internalType":"uint256","name":"iterations","type":"uint256"}],"name":"approximateDolaAndDbrNeeded","outputs":[{"internalType":"uint256","name":"dolaForDbr","type":"uint256"},{"internalType":"uint256","name":"dbrNeeded","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"contract IMarket","name":"market","type":"address"},{"internalType":"uint256","name":"dolaAmount","type":"uint256"},{"internalType":"uint256","name":"dolaForDbr","type":"uint256"},{"internalType":"uint256","name":"minDbr","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"buyDbrAndBorrowOnBehalf","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"curvePool","outputs":[{"internalType":"contract ICurvePool","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"contract IMarket","name":"market","type":"address"},{"internalType":"uint256","name":"collateralAmount","type":"uint256"},{"internalType":"uint256","name":"dolaAmount","type":"uint256"},{"internalType":"uint256","name":"dolaForDbr","type":"uint256"},{"internalType":"uint256","name":"minDbr","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"depositBuyDbrAndBorrowOnBehalf","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract IMarket","name":"market","type":"address"},{"internalType":"uint256","name":"borrowAmount","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"depositNativeEthAndBorrowOnBehalf","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"contract IMarket","name":"market","type":"address"},{"internalType":"uint256","name":"dolaAmount","type":"uint256"},{"internalType":"uint256","name":"dolaForDbr","type":"uint256"},{"internalType":"uint256","name":"minDbr","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"depositNativeEthBuyDbrAndBorrowOnBehalf","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"contract IMarket","name":"market","type":"address"}],"name":"depositNativeEthOnBehalf","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"contract IMarket","name":"market","type":"address"},{"internalType":"uint256","name":"dolaAmount","type":"uint256"},{"internalType":"uint256","name":"collateralAmount","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"repayAndWithdrawNativeEthOnBehalf","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract IMarket","name":"market","type":"address"},{"internalType":"uint256","name":"dolaAmount","type":"uint256"},{"internalType":"uint256","name":"minDolaFromDbr","type":"uint256"},{"internalType":"uint256","name":"dbrAmountToSell","type":"uint256"}],"name":"sellDbrAndRepayOnBehalf","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract IMarket","name":"market","type":"address"},{"internalType":"uint256","name":"dolaAmount","type":"uint256"},{"internalType":"uint256","name":"minDolaFromDbr","type":"uint256"},{"internalType":"uint256","name":"dbrAmountToSell","type":"uint256"},{"internalType":"uint256","name":"collateralAmount","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"sellDbrRepayAndWithdrawNativeEthOnBehalf","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract IMarket","name":"market","type":"address"},{"internalType":"uint256","name":"dolaAmount","type":"uint256"},{"internalType":"uint256","name":"minDolaFromDbr","type":"uint256"},{"internalType":"uint256","name":"dbrAmountToSell","type":"uint256"},{"internalType":"uint256","name":"collateralAmount","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"sellDbrRepayAndWithdrawOnBehalf","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract IMarket","name":"market","type":"address"},{"internalType":"uint256","name":"collateralAmount","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"withdrawNativeEthOnBehalf","outputs":[],"stateMutability":"nonpayable","type":"function"},{"stateMutability":"payable","type":"receive"}]