-
Notifications
You must be signed in to change notification settings - Fork 165
/
Copy pathMintVouchers.sol
54 lines (48 loc) · 2.16 KB
/
MintVouchers.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
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.23;
import "./ERC721.sol";
// An ERC721 whose owner can sign mint vouchers which people can redeem to mint
// tokens.
contract MintVouchersERC721 is ERC721 {
address payable immutable public owner;
mapping (bytes32 => bool) public isCancelled;
bytes32 immutable public DOMAIN_HASH;
bytes32 constant public VOUCHER_TYPE_HASH = keccak256('MintVoucher(uint256 tokenId,uint256 price)');
constructor() {
owner = payable(msg.sender);
DOMAIN_HASH = keccak256(abi.encode(
keccak256('EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)'),
keccak256('MintVouchersERC721'),
keccak256('1.0.0'),
block.chainid,
address(this)
));
}
// Pay for and claim a mint voucher in order to mint a token.
function mint(uint256 tokenId, uint8 v, bytes32 r, bytes32 s) external payable {
// Compute the EIP712 hash of the voucher.
bytes32 voucherHash = _getVoucherHash(tokenId, msg.value);
// Make sure the voucher hasn't been cancelled.
require(!isCancelled[voucherHash], 'mint voucher has been cancelled');
// Ensure that the owner signed the voucher.
require(ecrecover(voucherHash, v, r, s) == owner, 'invalid signature');
// No need to check price because if msg.value is not the price in the message
// the owner signed, the hash would be different and the signature check would
// fail.
_safeMint(tokenId, msg.sender); // Mint an NFT. Fails if already minted.
owner.transfer(msg.value); // Pay the owner.
}
// Cancel a mint voucher.
function cancel(uint256 tokenId, uint256 price) external {
require(msg.sender == owner, 'only owner');
isCancelled[_getVoucherHash(tokenId, price)] = true;
}
function _getVoucherHash(uint256 tokenId, uint256 price) internal view returns (bytes32) {
bytes32 structHash = keccak256(abi.encode(
VOUCHER_TYPE_HASH,
tokenId,
price
));
return keccak256(abi.encodePacked('\x19\x01', DOMAIN_HASH, structHash));
}
}