Skip to content

Commit

Permalink
Merge pull request #29 from pooltogether/fix/remove-shares-exploit-fix
Browse files Browse the repository at this point in the history
fix(contract): remove shares exploit fix
  • Loading branch information
PierrickGT authored Jul 7, 2022
2 parents 8928715 + 05938a7 commit d8843c1
Show file tree
Hide file tree
Showing 3 changed files with 5 additions and 37 deletions.
11 changes: 4 additions & 7 deletions contracts/yield-source/ATokenYieldSource.sol
Original file line number Diff line number Diff line change
Expand Up @@ -229,11 +229,10 @@ contract ATokenYieldSource is ERC20, IProtocolYieldSource, Manageable, Reentranc
uint256 shares = _tokenToShares(mintAmount);
_requireSharesGTZero(shares);

uint256 tokenAmount = _sharesToToken(shares);
_depositToAave(tokenAmount);
_depositToAave(mintAmount);
_mint(to, shares);

emit SuppliedTokenTo(msg.sender, shares, tokenAmount, to);
emit SuppliedTokenTo(msg.sender, shares, mintAmount, to);
}

/// @notice Redeems asset tokens from the yield source
Expand All @@ -245,19 +244,17 @@ contract ATokenYieldSource is ERC20, IProtocolYieldSource, Manageable, Reentranc
uint256 shares = _tokenToShares(redeemAmount);
_requireSharesGTZero(shares);

uint256 tokenAmount = _sharesToToken(shares);

_burn(msg.sender, shares);

IERC20 _depositToken = IERC20(_tokenAddress);
uint256 beforeBalance = _depositToken.balanceOf(address(this));
_lendingPool().withdraw(_tokenAddress, tokenAmount, address(this));
_lendingPool().withdraw(_tokenAddress, redeemAmount, address(this));
uint256 afterBalance = _depositToken.balanceOf(address(this));

uint256 balanceDiff = afterBalance.sub(beforeBalance);
_depositToken.safeTransfer(msg.sender, balanceDiff);

emit RedeemedToken(msg.sender, shares, tokenAmount);
emit RedeemedToken(msg.sender, shares, redeemAmount);
return balanceDiff;
}

Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@pooltogether/aave-yield-source",
"version": "1.2.0",
"version": "1.2.1",
"description": "PoolTogether Aave Yield Source",
"main": "index.js",
"license": "GPL-3.0",
Expand Down
29 changes: 0 additions & 29 deletions test/ATokenYieldSource.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -496,35 +496,6 @@ describe('ATokenYieldSource', () => {
aTokenYieldSource.connect(attacker).redeemToken(attackerRedeemAmount),
).to.be.revertedWith('ATokenYieldSource/shares-gt-zero');
});

it('should succeed to manipulate share price but fail to redeem more than deposited', async () => {
const amount = toWei('100000');
const attackAmount = BigNumber.from(1);
const aTokenAmount = toWei('10000');

await supplyTokenTo(attacker, attackAmount);

// Attacker sends 10000 aTokens directly to the contract to manipulate share price
await aToken.mint(attacker.address, aTokenAmount);
await aToken.connect(attacker).approve(aTokenYieldSource.address, aTokenAmount);
await aToken.connect(attacker).transfer(aTokenYieldSource.address, aTokenAmount);

await supplyTokenTo(wallet2, amount);

const sharePrice = await aTokenYieldSource.sharesToToken(BigNumber.from(1));

// Redeem 1 wei less than the full amount to burn 1 share instead of 2 because of rounding error
// The actual amount of shares to be burnt should be 1.99 but since Solidity truncates down, it will be 1
const attackerRedeemAmount = sharePrice.mul(2).sub(1);

const attackerRedeemShare = await aTokenYieldSource.tokenToShares(attackerRedeemAmount);
const redeemAmount = await aTokenYieldSource.sharesToToken(attackerRedeemShare);

await aTokenYieldSource.connect(attacker).redeemToken(attackerRedeemAmount);

expect(await usdcToken.balanceOf(attacker.address)).to.equal(redeemAmount);
expect(await aTokenYieldSource.balanceOfToken(attacker.address)).to.equal(Zero);
});
});

describe('transferERC20()', () => {
Expand Down

0 comments on commit d8843c1

Please sign in to comment.