diff --git a/contracts/adapters/paraswap/BaseParaSwapBuyAdapter.sol b/contracts/adapters/paraswap/BaseParaSwapBuyAdapter.sol index c6695837..48aea875 100644 --- a/contracts/adapters/paraswap/BaseParaSwapBuyAdapter.sol +++ b/contracts/adapters/paraswap/BaseParaSwapBuyAdapter.sol @@ -39,6 +39,7 @@ abstract contract BaseParaSwapBuyAdapter is BaseParaSwapAdapter { * @param maxAmountToSwap Max amount to be swapped * @param amountToReceive Amount to be received from the swap * @return amountSold The amount sold during the swap + * @return amountBought The amount bought during the swap */ function _buyOnParaSwap( uint256 toAmountOffset, @@ -47,7 +48,7 @@ abstract contract BaseParaSwapBuyAdapter is BaseParaSwapAdapter { IERC20Detailed assetToSwapTo, uint256 maxAmountToSwap, uint256 amountToReceive - ) internal returns (uint256 amountSold) { + ) internal returns (uint256 amountSold, uint256 amountBought) { (bytes memory buyCalldata, IParaSwapAugustus augustus) = abi.decode( paraswapData, (bytes, IParaSwapAugustus) @@ -75,7 +76,6 @@ abstract contract BaseParaSwapBuyAdapter is BaseParaSwapAdapter { uint256 balanceBeforeAssetTo = assetToSwapTo.balanceOf(address(this)); address tokenTransferProxy = augustus.getTokenTransferProxy(); - assetToSwapFrom.safeApprove(tokenTransferProxy, 0); assetToSwapFrom.safeApprove(tokenTransferProxy, maxAmountToSwap); if (toAmountOffset != 0) { @@ -101,12 +101,15 @@ abstract contract BaseParaSwapBuyAdapter is BaseParaSwapAdapter { } } + // Reset allowance + assetToSwapFrom.safeApprove(tokenTransferProxy, 0); + uint256 balanceAfterAssetFrom = assetToSwapFrom.balanceOf(address(this)); amountSold = balanceBeforeAssetFrom - balanceAfterAssetFrom; require(amountSold <= maxAmountToSwap, 'WRONG_BALANCE_AFTER_SWAP'); - uint256 amountReceived = assetToSwapTo.balanceOf(address(this)).sub(balanceBeforeAssetTo); - require(amountReceived >= amountToReceive, 'INSUFFICIENT_AMOUNT_RECEIVED'); + amountBought = assetToSwapTo.balanceOf(address(this)).sub(balanceBeforeAssetTo); + require(amountBought >= amountToReceive, 'INSUFFICIENT_AMOUNT_RECEIVED'); - emit Bought(address(assetToSwapFrom), address(assetToSwapTo), amountSold, amountReceived); + emit Bought(address(assetToSwapFrom), address(assetToSwapTo), amountSold, amountBought); } } diff --git a/contracts/adapters/paraswap/BaseParaSwapSellAdapter.sol b/contracts/adapters/paraswap/BaseParaSwapSellAdapter.sol index d6b76ced..4646e5a1 100644 --- a/contracts/adapters/paraswap/BaseParaSwapSellAdapter.sol +++ b/contracts/adapters/paraswap/BaseParaSwapSellAdapter.sol @@ -98,6 +98,7 @@ abstract contract BaseParaSwapSellAdapter is BaseParaSwapAdapter { revert(0, returndatasize()) } } + require( assetToSwapFrom.balanceOf(address(this)) == balanceBeforeAssetFrom - amountToSwap, 'WRONG_BALANCE_AFTER_SWAP' diff --git a/contracts/adapters/paraswap/ParaSwapRepayAdapter.sol b/contracts/adapters/paraswap/ParaSwapRepayAdapter.sol index 16039fcb..1e6a1569 100644 --- a/contracts/adapters/paraswap/ParaSwapRepayAdapter.sol +++ b/contracts/adapters/paraswap/ParaSwapRepayAdapter.sol @@ -114,7 +114,7 @@ contract ParaSwapRepayAdapter is BaseParaSwapBuyAdapter, ReentrancyGuard { // Pull aTokens from user _pullATokenAndWithdraw(address(collateralAsset), msg.sender, collateralAmount, permitSignature); //buy debt asset using collateral asset - uint256 amountSold = _buyOnParaSwap( + (uint256 amountSold, uint256 amountBought) = _buyOnParaSwap( buyAllBalanceOffset, paraswapData, collateralAsset, @@ -127,15 +127,23 @@ contract ParaSwapRepayAdapter is BaseParaSwapBuyAdapter, ReentrancyGuard { //deposit collateral back in the pool, if left after the swap(buy) if (collateralBalanceLeft > 0) { - IERC20(collateralAsset).safeApprove(address(POOL), 0); IERC20(collateralAsset).safeApprove(address(POOL), collateralBalanceLeft); POOL.deposit(address(collateralAsset), collateralBalanceLeft, msg.sender, 0); + IERC20(collateralAsset).safeApprove(address(POOL), 0); } // Repay debt. Approves 0 first to comply with tokens that implement the anti frontrunning approval fix - IERC20(debtAsset).safeApprove(address(POOL), 0); IERC20(debtAsset).safeApprove(address(POOL), debtRepayAmount); POOL.repay(address(debtAsset), debtRepayAmount, debtRateMode, msg.sender); + IERC20(debtAsset).safeApprove(address(POOL), 0); + + { + //transfer excess of debtAsset back to the user, if any + uint256 debtAssetExcess = amountBought - debtRepayAmount; + if (debtAssetExcess > 0) { + IERC20(debtAsset).safeTransfer(msg.sender, debtAssetExcess); + } + } } /** @@ -170,7 +178,7 @@ contract ParaSwapRepayAdapter is BaseParaSwapBuyAdapter, ReentrancyGuard { initiator ); - uint256 amountSold = _buyOnParaSwap( + (uint256 amountSold, uint256 amountBought) = _buyOnParaSwap( buyAllBalanceOffset, paraswapData, collateralAsset, @@ -180,9 +188,9 @@ contract ParaSwapRepayAdapter is BaseParaSwapBuyAdapter, ReentrancyGuard { ); // Repay debt. Approves for 0 first to comply with tokens that implement the anti frontrunning approval fix. - IERC20(debtAsset).safeApprove(address(POOL), 0); IERC20(debtAsset).safeApprove(address(POOL), debtRepayAmount); POOL.repay(address(debtAsset), debtRepayAmount, rateMode, initiator); + IERC20(debtAsset).safeApprove(address(POOL), 0); uint256 neededForFlashLoanRepay = amountSold.add(premium); @@ -194,6 +202,14 @@ contract ParaSwapRepayAdapter is BaseParaSwapBuyAdapter, ReentrancyGuard { permitSignature ); + { + //transfer excess of debtAsset back to the user, if any + uint256 debtAssetExcess = amountBought - debtRepayAmount; + if (debtAssetExcess > 0) { + IERC20(debtAsset).safeTransfer(initiator, debtAssetExcess); + } + } + // Repay flashloan. Approves for 0 first to comply with tokens that implement the anti frontrunning approval fix. IERC20(collateralAsset).safeApprove(address(POOL), 0); IERC20(collateralAsset).safeApprove(address(POOL), collateralAmount.add(premium)); diff --git a/contracts/mocks/swap/MockParaSwapAugustus.sol b/contracts/mocks/swap/MockParaSwapAugustus.sol index f05deca4..e56aaafc 100644 --- a/contracts/mocks/swap/MockParaSwapAugustus.sol +++ b/contracts/mocks/swap/MockParaSwapAugustus.sol @@ -20,6 +20,9 @@ contract MockParaSwapAugustus is IParaSwapAugustus { uint256 _expectedToAmountMax; uint256 _expectedToAmountMin; + uint256 _excessFromAmount; + uint256 _excessToAmount; + constructor() { TOKEN_TRANSFER_PROXY = new MockParaSwapTokenTransferProxy(); } @@ -58,6 +61,11 @@ contract MockParaSwapAugustus is IParaSwapAugustus { _expectedToAmountMax = toAmountMax; } + function expectExcess(uint256 excessFrom, uint256 excessTo) external { + _excessFromAmount = excessFrom; + _excessToAmount = excessTo; + } + function swap( address fromToken, address toToken, @@ -72,10 +80,17 @@ contract MockParaSwapAugustus is IParaSwapAugustus { 'From amount out of range' ); require(_receivedAmount >= toAmount, 'Received amount of tokens are less than expected'); - TOKEN_TRANSFER_PROXY.transferFrom(fromToken, msg.sender, address(this), fromAmount); - MintableERC20(toToken).mint(_receivedAmount); - IERC20(toToken).transfer(msg.sender, _receivedAmount); + TOKEN_TRANSFER_PROXY.transferFrom( + fromToken, + msg.sender, + address(this), + fromAmount - _excessFromAmount + ); + MintableERC20(toToken).mint(_receivedAmount + _excessToAmount); + IERC20(toToken).transfer(msg.sender, _receivedAmount + _excessToAmount); _expectingSwap = false; + _excessFromAmount = 0; + _excessToAmount = 0; return _receivedAmount; } @@ -93,10 +108,17 @@ contract MockParaSwapAugustus is IParaSwapAugustus { 'To amount out of range' ); require(_fromAmount <= fromAmount, 'From amount of tokens are higher than expected'); - TOKEN_TRANSFER_PROXY.transferFrom(fromToken, msg.sender, address(this), _fromAmount); - MintableERC20(toToken).mint(toAmount); - IERC20(toToken).transfer(msg.sender, toAmount); + TOKEN_TRANSFER_PROXY.transferFrom( + fromToken, + msg.sender, + address(this), + _fromAmount - _excessFromAmount + ); + MintableERC20(toToken).mint(toAmount + _excessToAmount); + IERC20(toToken).transfer(msg.sender, toAmount + _excessToAmount); _expectingSwap = false; + _excessFromAmount = 0; + _excessToAmount = 0; return fromAmount; } } diff --git a/test/paraswap/paraswapAdapters.repay.edge.spec.ts b/test/paraswap/paraswapAdapters.repay.edge.spec.ts new file mode 100644 index 00000000..99bfea8c --- /dev/null +++ b/test/paraswap/paraswapAdapters.repay.edge.spec.ts @@ -0,0 +1,325 @@ +import { + eContractid, + evmRevert, + evmSnapshot, + getContract, + getFirstSigner, + parseUnitsFromToken, + StableDebtToken, + tEthereumAddress, +} from '@aave/deploy-v3'; +import { + MockParaSwapAugustusRegistry__factory, + MockParaSwapAugustus__factory, + ParaSwapRepayAdapter, + ParaSwapRepayAdapter__factory, +} from '../../types'; +import { MockParaSwapAugustus } from '../../types/MockParaSwapAugustus'; +import { MockParaSwapAugustusRegistry } from '../../types/MockParaSwapAugustusRegistry'; +import { makeSuite, TestEnv } from '../helpers/make-suite'; +import { expect } from 'chai'; +import { parseEther } from 'ethers/lib/utils'; +import { buildParaswapBuyParams, buildParaSwapRepayParams } from './utils'; +import BigNumber from 'bignumber.js'; + +const EXCESS_FROM_AMOUNT = '11231'; +const EXCESS_TO_AMOUNT = '11231'; + +makeSuite('Paraswap adapters', (testEnv: TestEnv) => { + let mockAugustus: MockParaSwapAugustus; + let mockAugustusRegistry: MockParaSwapAugustusRegistry; + let paraswapRepayAdapter: ParaSwapRepayAdapter; + let evmSnapshotId: string; + + before(async () => { + const { addressesProvider, deployer } = testEnv; + + mockAugustus = await new MockParaSwapAugustus__factory(await getFirstSigner()).deploy(); + mockAugustusRegistry = await new MockParaSwapAugustusRegistry__factory( + await getFirstSigner() + ).deploy(mockAugustus.address); + paraswapRepayAdapter = await deployParaSwapRepayAdapter( + addressesProvider.address, + mockAugustusRegistry.address, + deployer.address + ); + }); + + beforeEach(async () => { + evmSnapshotId = await evmSnapshot(); + }); + + afterEach(async () => { + await evmRevert(evmSnapshotId); + }); + + describe('ParaswapRepayAdapter Edge', () => { + beforeEach(async () => { + const { users, weth, dai, usdc, aave, pool, deployer } = testEnv; + const userAddress = users[0].address; + + // Provide liquidity + await dai['mint(uint256)'](parseEther('400000')); + await dai.approve(pool.address, parseEther('400000')); + await pool.deposit(dai.address, parseEther('400000'), deployer.address, 0); + + const usdcLiquidity = await parseUnitsFromToken(usdc.address, '5000000'); + await usdc['mint(uint256)'](usdcLiquidity); + await usdc.approve(pool.address, usdcLiquidity); + await pool.deposit(usdc.address, usdcLiquidity, deployer.address, 0); + + await weth['mint(address,uint256)'](deployer.address, parseEther('100')); + await weth.approve(pool.address, parseEther('100')); + await pool.deposit(weth.address, parseEther('100'), deployer.address, 0); + + await aave['mint(uint256)'](parseEther('1000000')); + await aave.approve(pool.address, parseEther('1000000')); + await pool.deposit(aave.address, parseEther('1000000'), deployer.address, 0); + + // Make a deposit for user + await weth['mint(address,uint256)'](deployer.address, parseEther('1000')); + await weth.approve(pool.address, parseEther('1000')); + await pool.deposit(weth.address, parseEther('1000'), userAddress, 0); + + await aave['mint(uint256)'](parseEther('1000000')); + await aave.approve(pool.address, parseEther('1000000')); + await pool.deposit(aave.address, parseEther('1000000'), userAddress, 0); + + await usdc['mint(uint256)'](usdcLiquidity); + await usdc.approve(pool.address, usdcLiquidity); + await pool.deposit(usdc.address, usdcLiquidity, userAddress, 0); + }); + + describe('executeOperation', () => { + it('should swap, repay debt and pull the needed ATokens leaving no leftovers', async () => { + const { users, pool, weth, aWETH, oracle, dai, aDai, helpersContract } = testEnv; + const user = users[0].signer; + const userAddress = users[0].address; + + const amountWETHtoSwap = await parseUnitsFromToken(weth.address, '10'); + + const daiPrice = await oracle.getAssetPrice(dai.address); + const ethPrice = await oracle.getAssetPrice(weth.address); + const expectedDaiAmount = await parseUnitsFromToken( + dai.address, + new BigNumber(amountWETHtoSwap.toString()) + .times(ethPrice.toString()) + .div(daiPrice.toString()) + .shiftedBy(-18) + .toFixed(0) + ); + + // Open user Debt + await pool.connect(user).borrow(dai.address, expectedDaiAmount, 1, 0, userAddress); + + const daiStableDebtTokenAddress = ( + await helpersContract.getReserveTokensAddresses(dai.address) + ).stableDebtTokenAddress; + + const daiStableDebtContract = await getContract( + eContractid.StableDebtToken, + daiStableDebtTokenAddress + ); + + const userDaiStableDebtAmountBefore = await daiStableDebtContract.balanceOf(userAddress); + + await mockAugustus.expectBuy( + weth.address, + dai.address, + amountWETHtoSwap, + expectedDaiAmount, + expectedDaiAmount + ); + await mockAugustus.expectExcess(EXCESS_FROM_AMOUNT, EXCESS_TO_AMOUNT); + const mockAugustusCalldata = mockAugustus.interface.encodeFunctionData('buy', [ + weth.address, + dai.address, + amountWETHtoSwap, + expectedDaiAmount, + ]); + + const flashloanPremium = amountWETHtoSwap.mul(9).div(10000); + const flashloanTotal = amountWETHtoSwap.add(flashloanPremium); + await aWETH.connect(user).approve(paraswapRepayAdapter.address, flashloanTotal); + const userAEthBalanceBefore = await aWETH.balanceOf(userAddress); + + const params = buildParaSwapRepayParams( + dai.address, + expectedDaiAmount, + 0, + 1, + mockAugustusCalldata, + mockAugustus.address, + 0, + 0, + 0, + '0x0000000000000000000000000000000000000000000000000000000000000000', + '0x0000000000000000000000000000000000000000000000000000000000000000' + ); + + console.log(); + await expect( + pool + .connect(user) + .flashLoanSimple( + paraswapRepayAdapter.address, + weth.address, + amountWETHtoSwap.toString(), + params, + 0 + ) + ) + .to.emit(paraswapRepayAdapter, 'Bought') + .withArgs( + weth.address, + dai.address, + amountWETHtoSwap.sub(EXCESS_FROM_AMOUNT).toString(), + expectedDaiAmount.add(EXCESS_TO_AMOUNT) + ) + .to.emit(pool, 'Withdraw') + .withArgs( + weth.address, + paraswapRepayAdapter.address, + paraswapRepayAdapter.address, + flashloanTotal.sub(EXCESS_FROM_AMOUNT) + ); + + const adapterWethBalance = await weth.balanceOf(paraswapRepayAdapter.address); + const adapterDaiBalance = await dai.balanceOf(paraswapRepayAdapter.address); + const userDaiStableDebtAmount = await daiStableDebtContract.balanceOf(userAddress); + const userAEthBalance = await aWETH.balanceOf(userAddress); + const adapterAEthBalance = await aWETH.balanceOf(paraswapRepayAdapter.address); + const adapterADaiBalance = await aDai.balanceOf(paraswapRepayAdapter.address); + + expect(adapterAEthBalance).to.be.eq('0'); + expect(adapterADaiBalance).to.be.eq('0'); + expect(adapterWethBalance).to.be.eq('0'); + expect(adapterDaiBalance).to.be.eq('0'); + expect(userDaiStableDebtAmountBefore).to.be.gte(expectedDaiAmount); + expect(userDaiStableDebtAmount).to.be.lt(expectedDaiAmount); + expect(userAEthBalance).to.be.lt(userAEthBalanceBefore); + expect(userAEthBalance).to.be.gte( + userAEthBalanceBefore.sub(flashloanTotal).add(EXCESS_FROM_AMOUNT) + ); // gt because of interest + }); + }); + + describe('swapAndRepay', () => { + it('should correctly swap tokens and repay debt', async () => { + const { users, pool, weth, aWETH, oracle, dai, aDai, helpersContract } = testEnv; + const user = users[0].signer; + const userAddress = users[0].address; + + const amountWETHtoSwap = await parseUnitsFromToken(weth.address, '10'); + + const daiPrice = await oracle.getAssetPrice(dai.address); + const ethPrice = await oracle.getAssetPrice(weth.address); + const expectedDaiAmount = await parseUnitsFromToken( + dai.address, + new BigNumber(amountWETHtoSwap.toString()) + .times(ethPrice.toString()) + .div(daiPrice.toString()) + .shiftedBy(-18) + .toFixed(0) + ); + + // Open user Debt + await pool.connect(user).borrow(dai.address, expectedDaiAmount, 1, 0, userAddress); + + const daiStableDebtTokenAddress = ( + await helpersContract.getReserveTokensAddresses(dai.address) + ).stableDebtTokenAddress; + + const daiStableDebtContract = await getContract( + eContractid.StableDebtToken, + daiStableDebtTokenAddress + ); + + const userDaiStableDebtAmountBefore = await daiStableDebtContract.balanceOf(userAddress); + + const liquidityToSwap = amountWETHtoSwap; + const userAEthBalanceBefore = await aWETH.balanceOf(userAddress); + const userADaiBalanceBefore = await aDai.balanceOf(userAddress); + const userDaiBalanceBefore = await dai.balanceOf(userAddress); + + await mockAugustus.expectBuy( + weth.address, + dai.address, + liquidityToSwap, + expectedDaiAmount, + expectedDaiAmount + ); + await mockAugustus.expectExcess(EXCESS_FROM_AMOUNT, EXCESS_TO_AMOUNT); + const mockAugustusCalldata = mockAugustus.interface.encodeFunctionData('buy', [ + weth.address, + dai.address, + liquidityToSwap, + expectedDaiAmount, + ]); + + await aWETH.connect(user).approve(paraswapRepayAdapter.address, liquidityToSwap); + const params = buildParaswapBuyParams(mockAugustusCalldata, mockAugustus.address); + await expect( + paraswapRepayAdapter + .connect(user) + .swapAndRepay( + weth.address, + dai.address, + liquidityToSwap, + expectedDaiAmount, + 1, + 0, + params, + { + amount: 0, + deadline: 0, + v: 0, + r: '0x0000000000000000000000000000000000000000000000000000000000000000', + s: '0x0000000000000000000000000000000000000000000000000000000000000000', + } + ) + ) + .to.emit(paraswapRepayAdapter, 'Bought') + .withArgs( + weth.address, + dai.address, + liquidityToSwap.sub(EXCESS_FROM_AMOUNT).toString(), + expectedDaiAmount.add(EXCESS_TO_AMOUNT) + ); + + const adapterAEthBalance = await aWETH.balanceOf(paraswapRepayAdapter.address); + const adapterADaiBalance = await aDai.balanceOf(paraswapRepayAdapter.address); + const adapterWethBalance = await weth.balanceOf(paraswapRepayAdapter.address); + const adapterDaiBalance = await dai.balanceOf(paraswapRepayAdapter.address); + const userDaiStableDebtAmount = await daiStableDebtContract.balanceOf(userAddress); + const userDaiBalance = await dai.balanceOf(userAddress); + const userADaiBalance = await aDai.balanceOf(userAddress); + const userAEthBalance = await aWETH.balanceOf(userAddress); + + expect(adapterAEthBalance).to.be.eq('0'); + expect(adapterADaiBalance).to.be.eq('0'); + expect(adapterWethBalance).to.be.eq('0'); + expect(adapterDaiBalance).to.be.eq('0'); + expect(userDaiStableDebtAmountBefore).to.be.gte(expectedDaiAmount); + expect(userDaiStableDebtAmount).to.be.lt(expectedDaiAmount); + expect(userAEthBalance).to.be.eq( + userAEthBalanceBefore.sub(liquidityToSwap).add(EXCESS_FROM_AMOUNT) + ); + expect(userADaiBalance).to.be.eq(userADaiBalanceBefore); + expect(userDaiBalance).to.be.eq(userDaiBalanceBefore.add(EXCESS_TO_AMOUNT)); + }); + }); + }); +}); + +async function deployParaSwapRepayAdapter( + poolAddressesProvider: tEthereumAddress, + augustusRegistry: tEthereumAddress, + owner: tEthereumAddress +) { + return await new ParaSwapRepayAdapter__factory(await getFirstSigner()).deploy( + poolAddressesProvider, + augustusRegistry, + owner + ); +}