@@ -16,6 +16,7 @@ import "@solady/src/utils/MerkleProofLib.sol";
16
16
import "@solady/src/utils/ECDSA.sol " ;
17
17
import "@solady/src/utils/EIP712.sol " ;
18
18
import "@solady/src/utils/SafeTransferLib.sol " ;
19
+ import "@solady/src/utils/SignatureCheckerLib.sol " ;
19
20
20
21
import { Initializable } from "../../../extension/Initializable.sol " ;
21
22
import { Ownable } from "../../../extension/Ownable.sol " ;
@@ -25,8 +26,6 @@ import "../../../eip/interface/IERC721.sol";
25
26
import "../../../eip/interface/IERC1155.sol " ;
26
27
27
28
contract Airdrop is EIP712 , Initializable , Ownable {
28
- using ECDSA for bytes32 ;
29
-
30
29
/*///////////////////////////////////////////////////////////////
31
30
State, constants & structs
32
31
//////////////////////////////////////////////////////////////*/
@@ -38,7 +37,7 @@ contract Airdrop is EIP712, Initializable, Ownable {
38
37
/// @dev conditionId => hash(claimer address, token address, token id [1155]) => has claimed
39
38
mapping (uint256 => mapping (bytes32 => bool )) private claimed;
40
39
/// @dev Mapping from request UID => whether the request is processed.
41
- mapping (bytes32 => bool ) private processed;
40
+ mapping (bytes32 => bool ) public processed;
42
41
43
42
struct AirdropContentERC20 {
44
43
address recipient;
@@ -106,19 +105,18 @@ contract Airdrop is EIP712, Initializable, Ownable {
106
105
107
106
error AirdropInvalidProof ();
108
107
error AirdropAlreadyClaimed ();
109
- error AirdropFailed ();
110
108
error AirdropNoMerkleRoot ();
111
109
error AirdropValueMismatch ();
112
110
error AirdropRequestExpired (uint256 expirationTimestamp );
113
111
error AirdropRequestAlreadyProcessed ();
114
112
error AirdropRequestInvalidSigner ();
115
- error AirdropInvalidTokenAddress ();
116
113
117
114
/*///////////////////////////////////////////////////////////////
118
115
Events
119
116
//////////////////////////////////////////////////////////////*/
120
117
121
118
event Airdrop (address token );
119
+ event AirdropWithSignature (address token );
122
120
event AirdropClaimed (address token , address receiver );
123
121
124
122
/*///////////////////////////////////////////////////////////////
@@ -133,34 +131,25 @@ contract Airdrop is EIP712, Initializable, Ownable {
133
131
_setupOwner (_defaultAdmin);
134
132
}
135
133
136
- /*///////////////////////////////////////////////////////////////
137
- Receive and withdraw logic
138
- //////////////////////////////////////////////////////////////*/
139
-
140
- receive () external payable {}
141
-
142
- function withdraw (address _tokenAddress , uint256 _amount ) external onlyOwner {
143
- if (_tokenAddress == NATIVE_TOKEN_ADDRESS) {
144
- SafeTransferLib.safeTransferETH (msg .sender , _amount);
145
- } else {
146
- SafeTransferLib.safeTransferFrom (_tokenAddress, address (this ), msg .sender , _amount);
147
- }
148
- }
149
-
150
134
/*///////////////////////////////////////////////////////////////
151
135
Airdrop Push
152
136
//////////////////////////////////////////////////////////////*/
153
137
138
+ /**
139
+ * @notice Lets contract-owner send native token (eth) to a list of addresses.
140
+ * @dev Owner should send total airdrop amount as msg.value.
141
+ * Can only be called by contract owner.
142
+ *
143
+ * @param _contents List containing recipients and amounts to airdrop
144
+ */
154
145
function airdropNativeToken (AirdropContentERC20[] calldata _contents ) external payable onlyOwner {
155
146
uint256 len = _contents.length ;
156
147
uint256 nativeTokenAmount;
157
148
158
149
for (uint256 i = 0 ; i < len; i++ ) {
159
150
nativeTokenAmount += _contents[i].amount;
160
- (bool success , ) = _contents[i].recipient.call { value: _contents[i].amount }("" );
161
- if (! success) {
162
- revert AirdropFailed ();
163
- }
151
+
152
+ SafeTransferLib.safeTransferETH (_contents[i].recipient, _contents[i].amount);
164
153
}
165
154
166
155
if (nativeTokenAmount != msg .value ) {
@@ -170,6 +159,14 @@ contract Airdrop is EIP712, Initializable, Ownable {
170
159
emit Airdrop (NATIVE_TOKEN_ADDRESS);
171
160
}
172
161
162
+ /**
163
+ * @notice Lets contract owner send ERC20 tokens to a list of addresses.
164
+ * @dev The token-owner should approve total airdrop amount to this contract.
165
+ * Can only be called by contract owner.
166
+ *
167
+ * @param _tokenAddress Address of the ERC20 token being airdropped
168
+ * @param _contents List containing recipients and amounts to airdrop
169
+ */
173
170
function airdropERC20 (address _tokenAddress , AirdropContentERC20[] calldata _contents ) external onlyOwner {
174
171
uint256 len = _contents.length ;
175
172
@@ -180,6 +177,14 @@ contract Airdrop is EIP712, Initializable, Ownable {
180
177
emit Airdrop (_tokenAddress);
181
178
}
182
179
180
+ /**
181
+ * @notice Lets contract owner send ERC721 tokens to a list of addresses.
182
+ * @dev The token-owner should approve airdrop tokenIds to this contract.
183
+ * Can only be called by contract owner.
184
+ *
185
+ * @param _tokenAddress Address of the ERC721 token being airdropped
186
+ * @param _contents List containing recipients and tokenIds to airdrop
187
+ */
183
188
function airdropERC721 (address _tokenAddress , AirdropContentERC721[] calldata _contents ) external onlyOwner {
184
189
uint256 len = _contents.length ;
185
190
@@ -190,6 +195,14 @@ contract Airdrop is EIP712, Initializable, Ownable {
190
195
emit Airdrop (_tokenAddress);
191
196
}
192
197
198
+ /**
199
+ * @notice Lets contract owner send ERC1155 tokens to a list of addresses.
200
+ * @dev The token-owner should approve airdrop tokenIds and amounts to this contract.
201
+ * Can only be called by contract owner.
202
+ *
203
+ * @param _tokenAddress Address of the ERC1155 token being airdropped
204
+ * @param _contents List containing recipients, tokenIds, and amounts to airdrop
205
+ */
193
206
function airdropERC1155 (address _tokenAddress , AirdropContentERC1155[] calldata _contents ) external onlyOwner {
194
207
uint256 len = _contents.length ;
195
208
@@ -210,6 +223,14 @@ contract Airdrop is EIP712, Initializable, Ownable {
210
223
Airdrop With Signature
211
224
//////////////////////////////////////////////////////////////*/
212
225
226
+ /**
227
+ * @notice Lets contract owner send ERC20 tokens to a list of addresses with EIP-712 signature.
228
+ * @dev The token-owner should approve airdrop amounts to this contract.
229
+ * Signer should be the contract owner.
230
+ *
231
+ * @param req Struct containing airdrop contents, uid, and expiration timestamp
232
+ * @param signature EIP-712 signature to perform the airdrop
233
+ */
213
234
function airdropERC20WithSignature (AirdropRequestERC20 calldata req , bytes calldata signature ) external {
214
235
// verify expiration timestamp
215
236
if (req.expirationTimestamp < block .timestamp ) {
@@ -239,9 +260,17 @@ contract Airdrop is EIP712, Initializable, Ownable {
239
260
);
240
261
}
241
262
242
- emit Airdrop (req.tokenAddress);
263
+ emit AirdropWithSignature (req.tokenAddress);
243
264
}
244
265
266
+ /**
267
+ * @notice Lets contract owner send ERC721 tokens to a list of addresses with EIP-712 signature.
268
+ * @dev The token-owner should approve airdrop tokenIds to this contract.
269
+ * Signer should be the contract owner.
270
+ *
271
+ * @param req Struct containing airdrop contents, uid, and expiration timestamp
272
+ * @param signature EIP-712 signature to perform the airdrop
273
+ */
245
274
function airdropERC721WithSignature (AirdropRequestERC721 calldata req , bytes calldata signature ) external {
246
275
// verify expiration timestamp
247
276
if (req.expirationTimestamp < block .timestamp ) {
@@ -266,9 +295,17 @@ contract Airdrop is EIP712, Initializable, Ownable {
266
295
IERC721 (req.tokenAddress).safeTransferFrom (_from, req.contents[i].recipient, req.contents[i].tokenId);
267
296
}
268
297
269
- emit Airdrop (req.tokenAddress);
298
+ emit AirdropWithSignature (req.tokenAddress);
270
299
}
271
300
301
+ /**
302
+ * @notice Lets contract owner send ERC1155 tokens to a list of addresses with EIP-712 signature.
303
+ * @dev The token-owner should approve airdrop tokenIds and amounts to this contract.
304
+ * Signer should be the contract owner.
305
+ *
306
+ * @param req Struct containing airdrop contents, uid, and expiration timestamp
307
+ * @param signature EIP-712 signature to perform the airdrop
308
+ */
272
309
function airdropERC1155WithSignature (AirdropRequestERC1155 calldata req , bytes calldata signature ) external {
273
310
// verify expiration timestamp
274
311
if (req.expirationTimestamp < block .timestamp ) {
@@ -299,13 +336,23 @@ contract Airdrop is EIP712, Initializable, Ownable {
299
336
);
300
337
}
301
338
302
- emit Airdrop (req.tokenAddress);
339
+ emit AirdropWithSignature (req.tokenAddress);
303
340
}
304
341
305
342
/*///////////////////////////////////////////////////////////////
306
343
Airdrop Claimable
307
344
//////////////////////////////////////////////////////////////*/
308
345
346
+ /**
347
+ * @notice Lets allowlisted addresses claim ERC20 airdrop tokens.
348
+ * @dev The token-owner should approve total airdrop amount to this contract,
349
+ * and set merkle root of allowlisted address for this airdrop.
350
+ *
351
+ * @param _token Address of ERC20 airdrop token
352
+ * @param _receiver Allowlisted address for which the token is being claimed
353
+ * @param _quantity Allowlisted quantity of tokens to claim
354
+ * @param _proofs Merkle proofs for allowlist verification
355
+ */
309
356
function claimERC20 (address _token , address _receiver , uint256 _quantity , bytes32 [] calldata _proofs ) external {
310
357
bytes32 claimHash = _getClaimHashERC20 (_receiver, _token);
311
358
uint256 conditionId = tokenConditionId[_token];
@@ -319,7 +366,7 @@ contract Airdrop is EIP712, Initializable, Ownable {
319
366
revert AirdropNoMerkleRoot ();
320
367
}
321
368
322
- bool valid = MerkleProofLib.verify (
369
+ bool valid = MerkleProofLib.verifyCalldata (
323
370
_proofs,
324
371
_tokenMerkleRoot,
325
372
keccak256 (abi.encodePacked (_receiver, _quantity))
@@ -335,6 +382,16 @@ contract Airdrop is EIP712, Initializable, Ownable {
335
382
emit AirdropClaimed (_token, _receiver);
336
383
}
337
384
385
+ /**
386
+ * @notice Lets allowlisted addresses claim ERC721 airdrop tokens.
387
+ * @dev The token-owner should approve airdrop tokenIds to this contract,
388
+ * and set merkle root of allowlisted address for this airdrop.
389
+ *
390
+ * @param _token Address of ERC721 airdrop token
391
+ * @param _receiver Allowlisted address for which the token is being claimed
392
+ * @param _tokenId Allowlisted tokenId to claim
393
+ * @param _proofs Merkle proofs for allowlist verification
394
+ */
338
395
function claimERC721 (address _token , address _receiver , uint256 _tokenId , bytes32 [] calldata _proofs ) external {
339
396
bytes32 claimHash = _getClaimHashERC721 (_receiver, _token, _tokenId);
340
397
uint256 conditionId = tokenConditionId[_token];
@@ -348,7 +405,11 @@ contract Airdrop is EIP712, Initializable, Ownable {
348
405
revert AirdropNoMerkleRoot ();
349
406
}
350
407
351
- bool valid = MerkleProofLib.verify (_proofs, _tokenMerkleRoot, keccak256 (abi.encodePacked (_receiver, _tokenId)));
408
+ bool valid = MerkleProofLib.verifyCalldata (
409
+ _proofs,
410
+ _tokenMerkleRoot,
411
+ keccak256 (abi.encodePacked (_receiver, _tokenId))
412
+ );
352
413
if (! valid) {
353
414
revert AirdropInvalidProof ();
354
415
}
@@ -360,6 +421,17 @@ contract Airdrop is EIP712, Initializable, Ownable {
360
421
emit AirdropClaimed (_token, _receiver);
361
422
}
362
423
424
+ /**
425
+ * @notice Lets allowlisted addresses claim ERC1155 airdrop tokens.
426
+ * @dev The token-owner should approve tokenIds and total airdrop amounts to this contract,
427
+ * and set merkle root of allowlisted address for this airdrop.
428
+ *
429
+ * @param _token Address of ERC1155 airdrop token
430
+ * @param _receiver Allowlisted address for which the token is being claimed
431
+ * @param _tokenId Allowlisted tokenId to claim
432
+ * @param _quantity Allowlisted quantity of tokens to claim
433
+ * @param _proofs Merkle proofs for allowlist verification
434
+ */
363
435
function claimERC1155 (
364
436
address _token ,
365
437
address _receiver ,
@@ -379,7 +451,7 @@ contract Airdrop is EIP712, Initializable, Ownable {
379
451
revert AirdropNoMerkleRoot ();
380
452
}
381
453
382
- bool valid = MerkleProofLib.verify (
454
+ bool valid = MerkleProofLib.verifyCalldata (
383
455
_proofs,
384
456
_tokenMerkleRoot,
385
457
keccak256 (abi.encodePacked (_receiver, _tokenId, _quantity))
@@ -399,6 +471,13 @@ contract Airdrop is EIP712, Initializable, Ownable {
399
471
Setter functions
400
472
//////////////////////////////////////////////////////////////*/
401
473
474
+ /**
475
+ * @notice Lets contract owner set merkle root (allowlist) for claim based airdrops.
476
+ *
477
+ * @param _token Address of airdrop token
478
+ * @param _tokenMerkleRoot Merkle root of allowlist
479
+ * @param _resetClaimStatus Reset claim status / amount claimed so far to zero for all recipients
480
+ */
402
481
function setMerkleRoot (address _token , bytes32 _tokenMerkleRoot , bool _resetClaimStatus ) external onlyOwner {
403
482
if (_resetClaimStatus || tokenConditionId[_token] == 0 ) {
404
483
tokenConditionId[_token] += 1 ;
@@ -410,6 +489,7 @@ contract Airdrop is EIP712, Initializable, Ownable {
410
489
Miscellaneous
411
490
//////////////////////////////////////////////////////////////*/
412
491
492
+ /// @notice Returns claim status of a receiver for a claim based airdrop
413
493
function isClaimed (address _receiver , address _token , uint256 _tokenId ) external view returns (bool ) {
414
494
uint256 _conditionId = tokenConditionId[_token];
415
495
@@ -425,28 +505,33 @@ contract Airdrop is EIP712, Initializable, Ownable {
425
505
426
506
return false ;
427
507
}
428
-
508
+ /// @dev Checks whether contract owner can be set in the given execution context.
429
509
function _canSetOwner () internal view virtual override returns (bool ) {
430
510
return msg .sender == owner ();
431
511
}
432
512
513
+ /// @dev Domain name and version for EIP-712
433
514
function _domainNameAndVersion () internal pure override returns (string memory name , string memory version ) {
434
515
name = "Airdrop " ;
435
516
version = "1 " ;
436
517
}
437
518
519
+ /// @dev Keccak256 hash of receiver and token addresses, for claim based airdrop status tracking
438
520
function _getClaimHashERC20 (address _receiver , address _token ) private view returns (bytes32 ) {
439
521
return keccak256 (abi.encodePacked (_receiver, _token));
440
522
}
441
523
524
+ /// @dev Keccak256 hash of receiver, token address and tokenId, for claim based airdrop status tracking
442
525
function _getClaimHashERC721 (address _receiver , address _token , uint256 _tokenId ) private view returns (bytes32 ) {
443
526
return keccak256 (abi.encodePacked (_receiver, _token, _tokenId));
444
527
}
445
528
529
+ /// @dev Keccak256 hash of receiver, token address and tokenId, for claim based airdrop status tracking
446
530
function _getClaimHashERC1155 (address _receiver , address _token , uint256 _tokenId ) private view returns (bytes32 ) {
447
531
return keccak256 (abi.encodePacked (_receiver, _token, _tokenId));
448
532
}
449
533
534
+ /// @dev Hash nested struct within AirdropRequest___
450
535
function _hashContentInfoERC20 (AirdropContentERC20[] calldata contents ) private pure returns (bytes32 ) {
451
536
bytes32 [] memory contentHashes = new bytes32 [](contents.length );
452
537
for (uint256 i = 0 ; i < contents.length ; i++ ) {
@@ -455,6 +540,7 @@ contract Airdrop is EIP712, Initializable, Ownable {
455
540
return keccak256 (abi.encodePacked (contentHashes));
456
541
}
457
542
543
+ /// @dev Hash nested struct within AirdropRequest___
458
544
function _hashContentInfoERC721 (AirdropContentERC721[] calldata contents ) private pure returns (bytes32 ) {
459
545
bytes32 [] memory contentHashes = new bytes32 [](contents.length );
460
546
for (uint256 i = 0 ; i < contents.length ; i++ ) {
@@ -465,6 +551,7 @@ contract Airdrop is EIP712, Initializable, Ownable {
465
551
return keccak256 (abi.encodePacked (contentHashes));
466
552
}
467
553
554
+ /// @dev Hash nested struct within AirdropRequest___
468
555
function _hashContentInfoERC1155 (AirdropContentERC1155[] calldata contents ) private pure returns (bytes32 ) {
469
556
bytes32 [] memory contentHashes = new bytes32 [](contents.length );
470
557
for (uint256 i = 0 ; i < contents.length ; i++ ) {
@@ -475,6 +562,7 @@ contract Airdrop is EIP712, Initializable, Ownable {
475
562
return keccak256 (abi.encodePacked (contentHashes));
476
563
}
477
564
565
+ /// @dev Verify EIP-712 signature
478
566
function _verifyRequestSignerERC20 (
479
567
AirdropRequestERC20 calldata req ,
480
568
bytes calldata signature
@@ -485,10 +573,11 @@ contract Airdrop is EIP712, Initializable, Ownable {
485
573
);
486
574
487
575
bytes32 digest = _hashTypedData (structHash);
488
- address recovered = digest. recover (signature);
489
- return recovered == owner ();
576
+
577
+ return SignatureCheckerLib. isValidSignatureNowCalldata ( owner (), digest, signature );
490
578
}
491
579
580
+ /// @dev Verify EIP-712 signature
492
581
function _verifyRequestSignerERC721 (
493
582
AirdropRequestERC721 calldata req ,
494
583
bytes calldata signature
@@ -499,10 +588,11 @@ contract Airdrop is EIP712, Initializable, Ownable {
499
588
);
500
589
501
590
bytes32 digest = _hashTypedData (structHash);
502
- address recovered = digest. recover (signature);
503
- return recovered == owner ();
591
+
592
+ return SignatureCheckerLib. isValidSignatureNowCalldata ( owner (), digest, signature );
504
593
}
505
594
595
+ /// @dev Verify EIP-712 signature
506
596
function _verifyRequestSignerERC1155 (
507
597
AirdropRequestERC1155 calldata req ,
508
598
bytes calldata signature
@@ -513,7 +603,7 @@ contract Airdrop is EIP712, Initializable, Ownable {
513
603
);
514
604
515
605
bytes32 digest = _hashTypedData (structHash);
516
- address recovered = digest. recover (signature);
517
- return recovered == owner ();
606
+
607
+ return SignatureCheckerLib. isValidSignatureNowCalldata ( owner (), digest, signature );
518
608
}
519
609
}
0 commit comments