ERC20Permit.sol 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172
  1. // SPDX-License-Identifier: MIT
  2. // OpenZeppelin Contracts (last updated v4.9.0) (token/ERC20/extensions/ERC20Permit.sol)
  3. pragma solidity ^0.8.19;
  4. import "./IERC20Permit.sol";
  5. import "../ERC20.sol";
  6. import "../../../utils/cryptography/ECDSA.sol";
  7. import "../../../utils/cryptography/EIP712.sol";
  8. import "../../../utils/Nonces.sol";
  9. /**
  10. * @dev Implementation of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
  11. * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
  12. *
  13. * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
  14. * presenting a message signed by the account. By not relying on `{IERC20-approve}`, the token holder account doesn't
  15. * need to send a transaction, and thus is not required to hold Ether at all.
  16. *
  17. * _Available since v3.4._
  18. */
  19. abstract contract ERC20Permit is ERC20, IERC20Permit, EIP712, Nonces {
  20. // solhint-disable-next-line var-name-mixedcase
  21. bytes32 private constant _PERMIT_TYPEHASH =
  22. keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)");
  23. /**
  24. * @dev Initializes the {EIP712} domain separator using the `name` parameter, and setting `version` to `"1"`.
  25. *
  26. * It's a good idea to use the same `name` that is defined as the ERC20 token name.
  27. */
  28. constructor(string memory name) EIP712(name, "1") {}
  29. /**
  30. * @dev See {IERC20Permit-permit}.
  31. */
  32. function permit(
  33. address owner,
  34. address spender,
  35. uint256 value,
  36. uint256 deadline,
  37. uint8 v,
  38. bytes32 r,
  39. bytes32 s
  40. ) public virtual override {
  41. require(block.timestamp <= deadline, "ERC20Permit: expired deadline");
  42. bytes32 structHash = keccak256(abi.encode(_PERMIT_TYPEHASH, owner, spender, value, _useNonce(owner), deadline));
  43. bytes32 hash = _hashTypedDataV4(structHash);
  44. address signer = ECDSA.recover(hash, v, r, s);
  45. require(signer == owner, "ERC20Permit: invalid signature");
  46. _approve(owner, spender, value);
  47. }
  48. /**
  49. * @dev See {IERC20Permit-nonces}.
  50. */
  51. function nonces(address owner) public view virtual override(IERC20Permit, Nonces) returns (uint256) {
  52. return super.nonces(owner);
  53. }
  54. /**
  55. * @dev See {IERC20Permit-DOMAIN_SEPARATOR}.
  56. */
  57. // solhint-disable-next-line func-name-mixedcase
  58. function DOMAIN_SEPARATOR() external view virtual override returns (bytes32) {
  59. return _domainSeparatorV4();
  60. }
  61. }