123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778 |
- // SPDX-License-Identifier: MIT
- pragma solidity ^0.8.0;
- import "./draft-IERC20Permit.sol";
- import "../ERC20.sol";
- import "../../../utils/cryptography/draft-EIP712.sol";
- import "../../../utils/cryptography/ECDSA.sol";
- import "../../../utils/Counters.sol";
- /**
- * @dev Implementation of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
- * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
- *
- * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
- * presenting a message signed by the account. By not relying on `{IERC20-approve}`, the token holder account doesn't
- * need to send a transaction, and thus is not required to hold Ether at all.
- *
- * _Available since v3.4._
- */
- abstract contract ERC20Permit is ERC20, IERC20Permit, EIP712 {
- using Counters for Counters.Counter;
- mapping (address => Counters.Counter) private _nonces;
- // solhint-disable-next-line var-name-mixedcase
- bytes32 private immutable _PERMIT_TYPEHASH = keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)");
- /**
- * @dev Initializes the {EIP712} domain separator using the `name` parameter, and setting `version` to `"1"`.
- *
- * It's a good idea to use the same `name` that is defined as the ERC20 token name.
- */
- constructor(string memory name) EIP712(name, "1") {
- }
- /**
- * @dev See {IERC20Permit-permit}.
- */
- function permit(address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) public virtual override {
- // solhint-disable-next-line not-rely-on-time
- require(block.timestamp <= deadline, "ERC20Permit: expired deadline");
- bytes32 structHash = keccak256(
- abi.encode(
- _PERMIT_TYPEHASH,
- owner,
- spender,
- value,
- _nonces[owner].current(),
- deadline
- )
- );
- bytes32 hash = _hashTypedDataV4(structHash);
- address signer = ECDSA.recover(hash, v, r, s);
- require(signer == owner, "ERC20Permit: invalid signature");
- _nonces[owner].increment();
- _approve(owner, spender, value);
- }
- /**
- * @dev See {IERC20Permit-nonces}.
- */
- function nonces(address owner) public view override returns (uint256) {
- return _nonces[owner].current();
- }
- /**
- * @dev See {IERC20Permit-DOMAIN_SEPARATOR}.
- */
- // solhint-disable-next-line func-name-mixedcase
- function DOMAIN_SEPARATOR() external view override returns (bytes32) {
- return _domainSeparatorV4();
- }
- }
|