Liquidation bonus, which is supposed to incentivise liquidators to repay violators debt, is inflating the repayBorrowAmount instead of the seizeUnderlyingCollateralAmount breaking the functionality of the liquidation.
Vulnerability Details
The vulnerability occurs when a liquidator attempts to liquidate a borrower's debt by invoking the executeLiquidation function within the LoanManagerLogic contract. This function, in turn, calls the calcLiquidationAmounts function from LiquidationLogic to calculate the necessary transfers of collateral and borrowed amounts between the liquidator and the borrower. This function is supposed to take into consideration the liquidation bonus that the liquidator will get as an extra for liquidation and acts as the incentivise for him to perform it. We can see the implementation here of calcLiquidationAmounts here :
This function should correctly account for the liquidation bonus, which serves as an incentive for liquidators. However, in the case where seizeUnderlyingCollateralAmount > violatorUnderlingCollateralBalance, the repayBorrowAmount is recalculated and incorrectly inflated by the liquidation bonus. The correct behavior should be that the liquidation bonus inflates the seizeUnderlyingCollateralAmount, allowing the liquidator to seize additional collateral as a reward for performing the liquidation. Instead, the current implementation results in the liquidator being required to repay an inflated borrow amount, effectively causing the liquidator to pay more than the market value for the collateral. This misalignment of incentives not only discourages liquidators from executing liquidations but also breaks the fundamental logic of the liquidation process, leading to potential financial losses for the liquidator.
Impact Details
This vulnerability can lead to significant financial losses and operational inefficiencies within the system. Since the liquidation process is disincentivized, borrowers who fall below the required collateral thresholds may not be liquidated promptly, increasing systemic risk. Furthermore, liquidators who do engage in the process may suffer losses due to the inflated repayment amounts, which exceed the value of the collateral they seize. Generally speaking, the liquidation process is not working as expected.
To demonstrate the vulnerability, you can add the following test under the "Liquidate" section in LoanManager.test.ts and run npm test:
it.only("Should inflate incorrectly the repayBorrowAmount with the liquidation bonus instead of inflating the seizeCollateral",async () => {const {hub,loanManager,oracleManager,pools, loanId: violatorLoanId, accountId: violatorAccountId,loanTypeId,borrowAmount, usdcVariableInterestIndex: oldVariableInterestIndex, } =awaitloadFixture(depositEtherAndVariableBorrowUSDCFixture);// Config the liquidator.constliquidatorLoanId=getRandomBytes(BYTES32_LENGTH);constliquidatorAccountId=getAccountIdBytes("LIQUIDATOR_ACCOUNT_ID");constliquidatorLoanName=getRandomBytes(BYTES32_LENGTH);awaitloanManager.connect(hub).createUserLoan(liquidatorLoanId, liquidatorAccountId, loanTypeId, liquidatorLoanName);constliquidatorDepositAmount=BigInt(10000e6); // 10,000 USDCconstliquidatorDepositFAmount= liquidatorDepositAmount;constliquidatorDepositInterestIndex=BigInt(1e18);constusdcPrice=BigInt(1e18);awaitpools.USDC.pool.setDepositPoolParams({fAmount: liquidatorDepositFAmount,depositInterestIndex: liquidatorDepositInterestIndex,priceFeed: { price: usdcPrice, decimals:pools.USDC.tokenDecimals },});awaitloanManager.connect(hub).deposit(liquidatorLoanId, liquidatorAccountId,pools.USDC.poolId, liquidatorDepositAmount);// prepare liquidationconstethNodeOutputData=getNodeOutputData(BigInt(500e18));awaitoracleManager.setNodeOutput(pools.ETH.poolId,pools.ETH.tokenDecimals, ethNodeOutputData);// calculate interestconstvariableInterestIndex=BigInt(1.1e18);conststableInterestRate=BigInt(0.1e18);awaitpools.USDC.pool.setBorrowPoolParams({ variableInterestIndex, stableInterestRate });awaitpools.USDC.pool.setUpdatedVariableBorrowInterestIndex(variableInterestIndex);constborrowBalance=calcBorrowBalance(borrowAmount, variableInterestIndex, oldVariableInterestIndex);// Violator:// Collateral 1 ETH = $500// Borrow 1,000 USDC = $1,000// Liquidator:// Collateral 10,000 USDC = $10,000// Borrow $0constseizeCollateralAmount=BigInt(1e18); // 1 ETHconstseizeCollateralFAmount=BigInt(1e18);// The USDC that the liquidator will repay is 1 ETH in USDC + 4% bonus. He will repay 1.04 ETH in USDC.constrepayAmount=convToRepayBorrowAmount( seizeCollateralAmount,ethNodeOutputData.price,pools.ETH.tokenDecimals, usdcPrice,pools.USDC.tokenDecimals,BigInt(0.04e4) // liquidation bonus for usdc is 4% );constcollateralFAmount=convToCollateralFAmount( repayAmount,ethNodeOutputData.price,pools.ETH.tokenDecimals, usdcPrice,pools.USDC.tokenDecimals,BigInt(1e18) );constreserveCollateralFAmount=calcReserveCol( seizeCollateralFAmount, collateralFAmount,pools.ETH.liquidationFee );constliquidatorCollateralFAmount= seizeCollateralFAmount - reserveCollateralFAmount;constattemptedRepayAmount= repayAmount +BigInt(10e6);// Liquidation is executed.constminSeizedAmount=BigInt(0);constliquidate=await loanManager.connect(hub).liquidate( violatorLoanId, liquidatorLoanId, liquidatorAccountId,pools.ETH.poolId,pools.USDC.poolId, attemptedRepayAmount, minSeizedAmount );constborrowUSD= repayAmount * usdcPrice /BigInt(1e6); // USD value that the liquidator gained as debtconstcollateralUSD=toUnderlingAmount(liquidatorCollateralFAmount,BigInt(1e18)) *ethNodeOutputData.price /BigInt(1e18); // USD value that the liquidator gained as collateralconsole.log("borrowUSD",borrowUSD.toString());console.log("collateralUSD",collateralUSD.toString());// He got more debt than collateral, so clearly he is disentivised to do that.expect(borrowUSD > collateralUSD).to.be.true; });
This test will demonstrate that the liquidator ends up with more debt than the collateral they receive, highlighting the disincentive created by the vulnerability.