eip | title | author | discussions-to | status | type | category | created | requires |
---|---|---|---|---|---|---|---|---|
2280 |
erc-20 extension for native meta transactions support |
Iulian Rotaru (@mortimr) |
Draft |
Standards Track |
ERC |
2019-09-16 |
20, 165, 712 |
Use ERC-712 signatures to trigger transfer
, transferFrom
or approve
methods on an ERC-20 token.
The popularity of meta transactions is rising thanks to actors of the community (Austin Thomas Griffith, the MetaCartel, and many more). ERC-1776 was published in February 2019 by Ronan Sandford to start a standardization process around implementations of native meta transactions as extensions of well-known standards. I introduce today the specification for an ERC-20 (only) extension adding native meta transactions support; the mTKN
(meta token) standard. This standard should focus on simplicity, and is a major step in the development of standardized relay networks.
- Create a standard for wallets and web3 browsers to properly understand these types of signatures and adapt their UI
- Create a standard for Relays to properly integrate any new tokens following the standard
- Transfer Tokens without paying for ETH gas: this a better practice than giving eth to your users so they can make the transactions.
- Pay for gas with the token: the procotol supports a reward argument that redirects funds to the relayer when signature is properly uploaded.
- Make your tokens $ETH independent: Setup a network of relayers (accounts that want to submit signatures for you in exchange of the reward) and the token users will never have to buy ETH to use your tokens.
The following standard describe a set of functions to trigger the transfer
, approve
and transferFrom
mechanism with meta transactions. It introduces 7 new functions:
-
signedTransfer
& its verifierverifyTransfer
take the same arguments. The first one is a state-modifying method while the latter is a constant method meant for argument / signature verification. -
signedApprove
& its verifierverifyApprove
take the same arguments. The first one is a state-modifying method while the latter is a constant method meant for argument / signature verification. -
signedTransferFrom
& its verifierverifyTransferFrom
take the same arguments. The first one is a state-modifying method while the latter is a constant method meant for argument / signature verification. -
nonceOf
is a constant method used to retrieve the current nonce of an account.
The key words "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT", "SHOULD", "SHOULD NOT", "RECOMMENDED", "MAY", and "OPTIONAL" in this document are to be interpreted as described in RFC 2119.
This mandatory type MUST be included in the signed payload.
struct EIP712Domain {
string name;
string version;
uint256 chainId;
address verifyingContract;
}
This type describes the extra actors involved in the meta transaction: the Signer and the Relayer.
struct mActors {
address signer; // Used to check signature validity
address relayer; // Used to enforce a relayer address. Set to address(0) if no restrictions required
}
This type describes transaction parameters & reward parameters.
struct mTxParams {
uint256 nonce; // Nonce of the Signer
uint256 gasLimit; // Wanted gas limit
uint256 gasPrice; // Wanted gas price
uint256 reward; // Amount of token provided as reward to the relayer
}
This types describe the main type used to trigger a transfer
.
struct mTransfer {
address recipient; // Destination address of the transfer
uint256 amount; // Token amount to transfer
mActors actors;
mTxParams txparams;
}
This type describe the main type used to trigger an approve
.
struct mApprove {
address spender; // Target address that would benefit from the approval
uint256 amount; // Amount of token that the address can manipulate
mActors actors;
mTxParams txparams;
}
This type describe the main type used to trigger a transferFrom
.
struct mTransferFrom {
address sender; // Balance that transfers the tokens
address recipient; // Destination of the transfer
uint256 amount; // Amount to move from sender to recipient
mActors actors;
mTxParams txparams;
}
Name | bytes4 value |
---|---|
ERC-165 |
0x01ffc9a7 |
ERC-20 |
0x36372b07 |
ERC-20::name |
0x06fdde03 |
ERC-20::symbol |
0x95d89b41 |
ERC-20::decimals |
0x313ce567 |
ERC-2280 |
0x6941bcc3 |
function nonceOf(address account) public view returns (uint256);
The nonce is used to prevent attackers from organizing replay attacks.
Return the current expected nonce for given account
.
function verifyTransfer(
address recipient, uint256 amount,
address[2] memory actors, uint256[4] memory txparams, bytes memory signature
) public view returns (bool);
actors
is an array of address
es that MUST contain signer
as actors[0]
and relayer
as actors[1]
.
txparams
is an array of uint256
that MUST contain nonce
as txparams[0]
, gasLimit
as txparams[1]
, gasPrice
as txparams[2]
and reward
as txparams[3]
.
Verifies that a transfer to recipient
from signer
of amount
tokens is possible with the provided signature and with current contract state.
-
The function MUST throw if the
mTransfer
payload signature is invalid (resulting signer is different than providedsigner
). -
The function MUST throw if real
nonce
is different than thenonce
provided in themTransfer
payload. -
The function MUST throw if provided
gas
is not high enough to match thegasLimit
provided in themTransfer
payload. This should be checked as soon as the function starts (gasleft() >= gasLimit
). -
The function MUST throw if provided
gasPrice
is not high enough to match thegasLimit
provided in themTransfer
payload (tx.gasprice >= gasPrice
). -
The function MUST throw if provided
relayer
is notaddress(0)
ANDrelayer
is different thanmsg.sender
. -
The function MUST throw if the
signer
’s account balance does not have enough tokens to spend on transfer and on reward (balanceOf(signer) >= amount + reward
).
function signedTransfer(
address recipient, uint256 amount,
address[2] memory actors, uint256[4] memory txparams, bytes memory signature
) public returns (bool);
actors
is an array of address
es that MUST contain signer
as actors[0]
and relayer
as actors[1]
.
txparams
is an array of uint256
that MUST contain nonce
as txparams[0]
, gasLimit
as txparams[1]
, gasPrice
as txparams[2]
and reward
as txparams[3]
.
Transfers amount
amount of tokens from signer
's balance to address recipient
, and MUST fire the Transfer event.
-
The function MUST throw if the
mTransfer
payload signature is invalid (resulting signer is different than providedsigner
). -
The function MUST throw if real
nonce
is different than thenonce
provided in themTransfer
payload. -
The function MUST throw if provided
gas
is not high enough to match thegasLimit
provided in themTransfer
payload. This should be checked as soon as the function starts (gasleft() >= gasLimit
). -
The function MUST throw if provided
gasPrice
is not high enough to match thegasLimit
provided in themTransfer
payload (tx.gasprice >= gasPrice
), -
The function MUST throw if provided
relayer
is notaddress(0)
ANDrelayer
is different thanmsg.sender
. -
The function MUST throw if the
signer
’s account balance does not have enough tokens to spend on transfer and on reward (balanceOf(signer) >= amount + reward
).
function verifyApprove(
address spender, uint256 amount,
address[2] memory actors, uint256[4] memory txparams, bytes memory signature
) public view returns (bool);
actors
is an array of address
es that MUST contain signer
as actors[0]
and relayer
as actors[1]
.
txparams
is an array of uint256
that MUST contain nonce
as txparams[0]
, gasLimit
as txparams[1]
, gasPrice
as txparams[2]
and reward
as txparams[3]
.
Verifies that an approval for spender
of amount
tokens on signer
's balance is possible with the provided signature and with current contract state.
-
The function MUST throw if the
mApprove
payload signature is invalid (resulting signer is different than providedsigner
). -
The function MUST throw if real
nonce
is different than thenonce
provided in themApprove
payload. -
The function MUST throw if provided
gas
is not high enough to match thegasLimit
provided in themApprove
payload. This should be checked as soon as the function starts (gasleft() >= gasLimit
). -
The function MUST throw if provided
gasPrice
is not high enough to match thegasLimit
provided in themApprove
payload (tx.gasprice >= gasPrice
). -
The function MUST throw if provided
relayer
is notaddress(0)
ANDrelayer
is different thanmsg.sender
. -
The function MUST throw if the
signer
’s account balance does not have enough tokens to spend on allowance and on reward (balanceOf(signer) >= amount + reward
).
function signedApprove(
address spender, uint256 amount,
address[2] memory actors, uint256[4] memory txparams, bytes memory signature
) public returns (bool);
actors
is an array of address
es that MUST contain signer
as actors[0]
and relayer
as actors[1]
.
txparams
is an array of uint256
that MUST contain nonce
as txparams[0]
, gasLimit
as txparams[1]
, gasPrice
as txparams[2]
and reward
as txparams[3]
.
Approves amount
amount of tokens from signer
's balance to address spender
, and MUST fire the Approve event.
-
The function MUST throw if the
mApprove
payload signature is invalid (resulting signer is different than providedsigner
). -
The function MUST throw if real
nonce
is different than thenonce
provided in themApprove
payload. -
The function MUST throw if provided
gas
is not high enough to match thegasLimit
provided in themApprove
payload. This should be checked as soon as the function starts (gasleft() >= gasLimit
). -
The function MUST throw if provided
gasPrice
is not high enough to match thegasLimit
provided in themApprove
payload (tx.gasprice >= gasPrice
). -
The function MUST throw if provided
relayer
is notaddress(0)
ANDrelayer
is different thanmsg.sender
. -
The function MUST throw if the
signer
’s account balance does not have enough tokens to spend on allowance and on reward (balanceOf(signer) >= amount + reward
).
function verifyTransferFrom(
address sender, address recipient, uint256 amount,
address[2] memory actors, uint256[4] memory txparams, bytes memory signature
) public view returns (bool);
actors
is an array of address
es that MUST contain signer
as actors[0]
and relayer
as actors[1]
.
txparams
is an array of uint256
that MUST contain nonce
as txparams[0]
, gasLimit
as txparams[1]
, gasPrice
as txparams[2]
and reward
as txparams[3]
.
Verifies that a transfer from sender
to recipient
of amount
tokens and that signer
has at least amount
allowance from sender
is possible with the provided signature and with current contract state.
-
The function MUST throw if the
mTransferFrom
payload signature is invalid (resulting signer is different than providedsigner
). -
The function MUST throw if real
nonce
is different than thenonce
provided in themTransferFrom
payload. -
The function MUST throw if provided
gas
is not high enough to match thegasLimit
provided in themTransferFrom
payload. This should be checked as soon as the function starts (gasleft() >= gasLimit
). -
The function MUST throw if provided
gasPrice
is not high enough to match thegasLimit
provided in themTransferFrom
payload (tx.gasprice >= gasPrice
). -
The function MUST throw if provided
relayer
is notaddress(0)
ANDrelayer
is different thanmsg.sender
. -
The function SHOULD throw if the
signer
’s account balance does not have enough tokens to spend on reward (balanceOf(signer) >= reward
). -
The function MUST throw if the
signer
’s account allowance fromsender
is at leastamount
(allowance(sender, signer) >= amount
).
function signedTransferFrom(
address sender, address recipient, uint256 amount,
address[2] memory actors, uint256[4] memory txparams, bytes memory signature
) public returns (bool);
}
actors
is an array of address
es that MUST contain signer
as actors[0]
and relayer
as actors[1]
.
txparams
is an array of uint256
that MUST contain nonce
as txparams[0]
, gasLimit
as txparams[1]
, gasPrice
as txparams[2]
and reward
as txparams[3]
.
Triggers transfer from sender
to recipient
of amount
tokens. signer
MUST have at least amount
allowance from sender
. It MUST trigger a Transfer event.
-
The function MUST throw if the
mTransferFrom
payload signature is invalid (resulting signer is different than providedsigner
). -
The function MUST throw if real
nonce
is different than thenonce
provided in themTransferFrom
payload. -
The function MUST throw if provided
gas
is not high enough to match thegasLimit
provided in themTransferFrom
payload. This should be checked as soon as the function starts (gasleft() >= gasLimit
). -
The function MUST throw if provided
gasPrice
is not high enough to match thegasLimit
provided in themTransferFrom
payload (tx.gasprice >= gasPrice
). -
The function MUST throw if provided
relayer
is notaddress(0)
ANDrelayer
is different thanmsg.sender
. -
The function MUST throw if the
signer
’s account balance does not have enough tokens to spend on reward (balanceOf(signer) >= reward
). -
The function MUST throw if the
signer
’s account allowance fromsender
is at leastamount
(allowance(sender, signer) >= amount
).
The specification was designed with simplicity in mind, and respect to current tools and standards (EIP-712, ERC-165). The 7 new methods can be divided into 4 categories.
-
nonceOf
is the newcomer. This constant method is required to give a nonce to the signers and prevent replay attacks. -
verifyTransfer
andsignedTransfer
accept two arguments like the initialtransfer
method, plus theactors
,txparams
andsignature
required for the meta transaction. -
verifyApprove
andsignedApprove
accept two arguments like the initialapprove
method, plus theactors
,txparams
andsignature
required for the meta transaction. -
verifyTransferFrom
andsignedTransferFrom
accept three arguments like the initialtransferFrom
method, plus theactors
,txparams
andsignature
required for the meta transaction.
The mActors
and mTxParams
type have been introduced to factorize arguments in two logical types. Multiple meta-transactions could get packed into the arrays in that way.
The signature
is a bytes
argument, making it easier to use as most off-chain tools and implementations return hex strings or buffers as results of ECDSA.
The following interface can be found here.
pragma solidity >=0.5.0 <0.6.0;
/// @title ERC-2280 mTKN - ERC-20 native meta transactions
/// @dev See
interface mTKN {
/// @notice Return the current expected nonce for given `account`.
///
/// @param account Will retrieve the nonce of this address
///
/// @return The current nonce for `account`
///
function nonceOf(address account) public view returns (uint256);
/// @notice Verifies that a transfer to `recipient` from `signer` of `amount` tokens
/// is possible with the provided signature and with current contract state.
///
/// @dev The function MUST throw if the `mTransfer` payload signature is
/// invalid (resulting signer is different than provided `signer`).
///
/// @dev The function MUST throw if real `nonce` is not high enough to
/// match the `nonce` provided in the `mTransfer` payload.
///
/// @dev The function MUST throw if provided `gas` is not high enough
/// to match the `gasLimit` provided in the `mTransfer` payload.
/// This should be checked as soon as the function starts. (`gasleft() >= gasLimit`)
///
/// @dev The function MUST throw if provided `gasPrice` is not high enough
/// to match the `gasLimit` provided in the `mTransfer` payload. (`tx.gasprice >= gasPrice`)
///
/// @dev The function MUST throw if provided `relayer` is not `address(0)` AND `relayer`
/// is different than `msg.sender`.
///
/// @dev The function SHOULD throw if the `signer`’s account balance does not have enough
/// tokens to spend on transfer and on reward (`balanceOf(signer) >= amount + reward`).
///
/// @param recipient Target address of the transfer
/// @param amount Amount of token to transfer from `signer`'s balance to `recipient`'s balance
/// @param actors Array of `address`es that contains `signer` as `actors[0]` and `relayer` as `actors[1]` in this
/// precise order.
/// @param txparams Array of `uint256` that MUST contain `nonce` as `txparams[0]`, `gasLimit` as `txparams[1]`,
/// `gasPrice` as `txparams[2]` and `reward` as `txparams[3]` in this precise order.
///
function verifyTransfer(address recipient, uint256 amount, address[2] memory actors, uint256[4] memory txparams, bytes memory signature) external view returns (bool);
/// @notice Transfers `amount` amount of tokens to address `recipient`, and fires the Transfer event.
///
/// @dev The function MUST throw if the `mTransfer` payload signature is
/// invalid (resulting signer is different than provided `signer`).
///
/// @dev The function MUST throw if real `nonce` is not high enough to
/// match the `nonce` provided in the `mTransfer` payload.
///
/// @dev The function MUST throw if provided `gas` is not high enough
/// to match the `gasLimit` provided in the `mTransfer` payload.
/// This should be checked as soon as the function starts. (`gasleft() >= gasLimit`)
///
/// @dev The function MUST throw if provided `gasPrice` is not high enough
/// to match the `gasLimit` provided in the `mTransfer` payload. (`tx.gasprice >= gasPrice`)
///
/// @dev The function MUST throw if provided `relayer` is not `address(0)` AND `relayer`
/// is different than `msg.sender`.
///
/// @dev The function SHOULD throw if the `signer`’s account balance does not have enough
/// tokens to spend on transfer and on reward (`balanceOf(signer) >= amount + reward`).
///
/// @param recipient Target address of the transfer
/// @param amount Amount of token to transfer from `signer`'s balance to `recipient`'s balance
/// @param actors Array of `address`es that contains `signer` as `actors[0]` and `relayer` as `actors[1]` in this
/// precise order.
/// @param txparams Array of `uint256` that MUST contain `nonce` as `txparams[0]`, `gasLimit` as `txparams[1]`,
/// `gasPrice` as `txparams[2]` and `reward` as `txparams[3]` in this precise order.
///
function signedTransfer(address recipient, uint256 amount, address[2] memory actors, uint256[4] memory txparams, bytes memory signature) public returns (bool);
/// @notice Verifies that an approval for `spender` of `amount` tokens on
/// `signer`'s balance is possible with the provided signature and with current contract state.
///
/// @dev The function MUST throw if the `mTransfer` payload signature is
/// invalid (resulting signer is different than provided `signer`).
///
/// @dev The function MUST throw if real `nonce` is not high enough to
/// match the `nonce` provided in the `mTransfer` payload.
///
/// @dev The function MUST throw if provided `gas` is not high enough
/// to match the `gasLimit` provided in the `mTransfer` payload.
/// This should be checked as soon as the function starts. (`gasleft() >= gasLimit`)
///
/// @dev The function MUST throw if provided `gasPrice` is not high enough
/// to match the `gasLimit` provided in the `mTransfer` payload. (`tx.gasprice >= gasPrice`)
///
/// @dev The function MUST throw if provided `relayer` is not `address(0)` AND `relayer`
/// is different than `msg.sender`.
///
/// @dev The function SHOULD throw if the `signer`’s account balance does not have enough tokens
/// to spend on allowance and on reward (`balanceOf(signer) >= amount + reward`).
///
/// @param spender Target address of the approval
/// @param amount Amount of token to approve from `signer`'s balance to `recipient`'s account
/// @param actors Array of `address`es that contains `signer` as `actors[0]` and `relayer` as `actors[1]` in this
/// precise order.
/// @param txparams Array of `uint256` that MUST contain `nonce` as `txparams[0]`, `gasLimit` as `txparams[1]`,
/// `gasPrice` as `txparams[2]` and `reward` as `txparams[3]` in this precise order.
///
function verifyApprove(address spender, uint256 amount, address[2] memory actors, uint256[4] memory txparams, bytes memory signature) public view returns (bool);
/// @notice Approves `amount` amount of tokens from `signer`'s balance to address `spender`, and
/// MUST fire the Approve event.
///
/// @dev The function MUST throw if the `mTransfer` payload signature is
/// invalid (resulting signer is different than provided `signer`).
///
/// @dev The function MUST throw if real `nonce` is not high enough to
/// match the `nonce` provided in the `mTransfer` payload.
///
/// @dev The function MUST throw if provided `gas` is not high enough
/// to match the `gasLimit` provided in the `mTransfer` payload.
/// This should be checked as soon as the function starts. (`gasleft() >= gasLimit`)
///
/// @dev The function MUST throw if provided `gasPrice` is not high enough
/// to match the `gasLimit` provided in the `mTransfer` payload. (`tx.gasprice >= gasPrice`)
///
/// @dev The function MUST throw if provided `relayer` is not `address(0)` AND `relayer`
/// is different than `msg.sender`.
///
/// @dev The function SHOULD throw if the `signer`’s account balance does not have enough tokens
/// to spend on allowance and on reward (`balanceOf(signer) >= amount + reward`).
///
/// @param spender Target address of the approval
/// @param amount Amount of token to approve from `signer`'s balance to `recipient`'s account
/// @param actors Array of `address`es that contains `signer` as `actors[0]` and `relayer` as `actors[1]` in this
/// precise order.
/// @param txparams Array of `uint256` that MUST contain `nonce` as `txparams[0]`, `gasLimit` as `txparams[1]`,
/// `gasPrice` as `txparams[2]` and `reward` as `txparams[3]` in this precise order.
///
function signedApprove(address spender, uint256 amount, address[2] memory actors, uint256[4] memory txparams, bytes memory signature) public returns (bool);
/// @notice Verifies that a transfer from `sender` to `recipient` of `amount` tokens and that
/// `signer` has at least `amount` allowance from `sender` is possible with the
/// provided signature and with current contract state.
///
/// @dev The function MUST throw if the `mTransfer` payload signature is
/// invalid (resulting signer is different than provided `signer`).
///
/// @dev The function MUST throw if real `nonce` is not high enough to
/// match the `nonce` provided in the `mTransfer` payload.
///
/// @dev The function MUST throw if provided `gas` is not high enough
/// to match the `gasLimit` provided in the `mTransfer` payload.
/// This should be checked as soon as the function starts. (`gasleft() >= gasLimit`)
///
/// @dev The function MUST throw if provided `gasPrice` is not high enough
/// to match the `gasLimit` provided in the `mTransfer` payload. (`tx.gasprice >= gasPrice`)
///
/// @dev The function MUST throw if provided `relayer` is not `address(0)` AND `relayer`
/// is different than `msg.sender`.
///
/// @dev The function SHOULD throw if the `signer`’s account balance does not have enough tokens to spend
/// on reward (`balanceOf(signer) >= reward`).
///
/// @dev The function SHOULD throw if the `signer`’s account allowance from `sender` is at least `amount`
/// (`allowance(sender, signer) >= amount`).
///
/// @param sender Account that is send tokens
/// @param recipient Account that is receiving the tokens
/// @param amount Amount of token to transfer from `sender` to `recipient`. `signer` should have at least
/// least `amount` allowance from `sender`.
/// @param actors Array of `address`es that contains `signer` as `actors[0]` and `relayer` as `actors[1]` in this
/// precise order.
/// @param txparams Array of `uint256` that MUST contain `nonce` as `txparams[0]`, `gasLimit` as `txparams[1]`,
/// `gasPrice` as `txparams[2]` and `reward` as `txparams[3]` in this precise order.
///
function verifyTransferFrom(address sender, address recipient, uint256 amount, address[2] memory actors, uint256[4] memory txparams, bytes memory signature) public view returns (bool);
/// @notice Triggers transfer from `sender` to `recipient` of `amount` tokens. `signer`
/// MUST have at least `amount` allowance from `sender`.
/// It MUST trigger a Transfer event.
///
/// @dev The function MUST throw if the `mTransfer` payload signature is
/// invalid (resulting signer is different than provided `signer`).
///
/// @dev The function MUST throw if real `nonce` is not high enough to
/// match the `nonce` provided in the `mTransfer` payload.
///
/// @dev The function MUST throw if provided `gas` is not high enough
/// to match the `gasLimit` provided in the `mTransfer` payload.
/// This should be checked as soon as the function starts. (`gasleft() >= gasLimit`)
///
/// @dev The function MUST throw if provided `gasPrice` is not high enough
/// to match the `gasLimit` provided in the `mTransfer` payload. (`tx.gasprice >= gasPrice`)
///
/// @dev The function MUST throw if provided `relayer` is not `address(0)` AND `relayer`
/// is different than `msg.sender`.
///
/// @dev The function SHOULD throw if the `signer`’s account balance does not have enough tokens to spend
/// on reward (`balanceOf(signer) >= reward`).
///
/// @dev The function SHOULD throw if the `signer`’s account allowance from `sender` is at least `amount`
/// (`allowance(sender, signer) >= amount`).
///
/// @param sender Account that is send tokens
/// @param recipient Account that is receiving the tokens
/// @param amount Amount of token to transfer from `sender` to `recipient`. `signer` should have at least
/// least `amount` allowance from `sender`.
/// @param actors Array of `address`es that contains `signer` as `actors[0]` and `relayer` as `actors[1]` in this
/// precise order.
/// @param txparams Array of `uint256` that MUST contain `nonce` as `txparams[0]`, `gasLimit` as `txparams[1]`,
/// `gasPrice` as `txparams[2]` and `reward` as `txparams[3]` in this precise order.
///
function signedTransferFrom(address sender, address recipient, uint256 amount, address[2] memory actors, uint256[4] memory txparams, bytes memory signature) public returns (bool);
}
To lower the amount of arguments for the call, we prefer taking the signature as a bytes
argument. This functions splits the given bytes
into uint8 v
, bytes32 r
and bytes32 s
.
function _splitSignature(bytes memory signature) private pure returns (uint8 v, bytes32 r, bytes32 s) {
require(signature.length == 65, "Invalid signature length");
assembly {
r := mload(add(signature, 32))
s := mload(add(signature, 64))
v := and(mload(add(signature, 65)), 255)
}
if (v < 27) {
v += 27;
}
require(v == 27 || v == 28, "Invalid v argument");
}
In order to verify the signatures, the mTKN
compliant contract should use the following method:
- Define the types as structs (
EIP712Domain
,mActors
,mTxParams
,mTransfer
,mApprove
,mTransferFrom
). - Define a
hash
helper function with parametric polymorphism for each type defined above. - Define a
verify
helper function with parametric polymorphism for each type defined above that is going to be signed (onlymTransfer
,mApprove
andmTransferFrom
).
The EIP712Domain should always have the following values for your implementation:
name
should be the same as the one returned by thename
method from theERC20
optional specification.version
should be defined as you want.chainId
should be the proper ID of the current chain.verifyingContract
should be the address of themTKN
contract (address(this)
).
This is an example implementation of the mTKNDomain
, that can also be found here.
pragma solidity >=0.5.0 <0.6.0;
contract mTKNDomain {
struct Signature {
uint8 v;
bytes32 r;
bytes32 s;
}
struct EIP712Domain {
string name;
string version;
uint256 chainId;
address verifyingContract;
}
struct mTxParams {
uint256 nonce;
uint256 gasLimit;
uint256 gasPrice;
uint256 reward;
}
struct mActors {
address signer;
address relayer;
}
struct mTransfer {
address recipient;
uint256 amount;
mActors actors;
mTxParams txparams;
}
struct mApprove {
address spender;
uint256 amount;
mActors actors;
mTxParams txparams;
}
struct mTransferFrom {
address sender;
address recipient;
uint256 amount;
mActors actors;
mTxParams txparams;
}
bytes32 constant MACTORS_TYPEHASH = keccak256(
"mActors(address signer,address relayer)"
);
bytes32 constant MTXPARAMS_TYPEHASH = keccak256(
"mTxParams(uint256 nonce,uint256 gasLimit,uint256 gasPrice,uint256 reward)"
);
bytes32 constant EIP712DOMAIN_TYPEHASH = keccak256(
"EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"
);
bytes32 constant MTRANSFER_TYPEHASH = keccak256(
"mTransfer(address recipient,uint256 amount,mActors actors,mTxParams txparams)mActors(address signer,address relayer)mTxParams(uint256 nonce,uint256 gasLimit,uint256 gasPrice,uint256 reward)"
);
bytes32 constant MAPPROVE_TYPEHASH = keccak256(
"mApprove(address spender,uint256 amount,mActors actors,mTxParams txparams)mActors(address signer,address relayer)mTxParams(uint256 nonce,uint256 gasLimit,uint256 gasPrice,uint256 reward)"
);
bytes32 constant MTRANSFERFROM_TYPEHASH = keccak256(
"mTransferFrom(address sender,address recipient,uint256 amount,mActors actors,mTxParams txparams)mActors(address signer,address relayer)mTxParams(uint256 nonce,uint256 gasLimit,uint256 gasPrice,uint256 reward)"
);
bytes32 DOMAIN_SEPARATOR;
constructor (string memory domain_name) public {
DOMAIN_SEPARATOR = hash(EIP712Domain({
name: domain_name,
version: '1',
chainId: 1,
verifyingContract: address(this)
}));
}
function hash(EIP712Domain memory eip712Domain) internal pure returns (bytes32) {
return keccak256(abi.encode(
EIP712DOMAIN_TYPEHASH,
keccak256(bytes(eip712Domain.name)),
keccak256(bytes(eip712Domain.version)),
eip712Domain.chainId,
eip712Domain.verifyingContract
));
}
function hash(mTransfer memory transfer) internal pure returns (bytes32) {
return keccak256(abi.encode(
MTRANSFER_TYPEHASH,
transfer.recipient,
transfer.amount,
hash(transfer.actors),
hash(transfer.txparams)
));
}
function hash(mApprove memory approve) internal pure returns (bytes32) {
return keccak256(abi.encode(
MAPPROVE_TYPEHASH,
approve.spender,
approve.amount,
hash(approve.actors),
hash(approve.txparams)
));
}
function hash(mActors memory actors) internal pure returns (bytes32) {
return keccak256(abi.encode(
MACTORS_TYPEHASH,
actors.signer,
actors.relayer
));
}
function hash(mTxParams memory txparams) internal pure returns (bytes32) {
return keccak256(abi.encode(
MTXPARAMS_TYPEHASH,
txparams.nonce,
txparams.gasLimit,
txparams.gasPrice,
txparams.reward
));
}
function hash(mTransferFrom memory transfer_from) internal pure returns (bytes32) {
return keccak256(abi.encode(
MTRANSFERFROM_TYPEHASH,
transfer_from.sender,
transfer_from.recipient,
transfer_from.amount,
hash(transfer_from.actors),
hash(transfer_from.txparams)
));
}
function verify(mTransfer memory transfer, Signature memory signature) internal view returns (bool) {
bytes32 digest = keccak256(abi.encodePacked(
"\x19\x01",
DOMAIN_SEPARATOR,
hash(transfer)
));
return ecrecover(digest, signature.v, signature.r, signature.s) == transfer.actors.signer;
}
function verify(mApprove memory approve, Signature memory signature) internal view returns (bool) {
bytes32 digest = keccak256(abi.encodePacked(
"\x19\x01",
DOMAIN_SEPARATOR,
hash(approve)
));
return ecrecover(digest, signature.v, signature.r, signature.s) == approve.actors.signer;
}
function verify(mTransferFrom memory transfer_from, Signature memory signature) internal view returns (bool) {
bytes32 digest = keccak256(abi.encodePacked(
"\x19\x01",
DOMAIN_SEPARATOR,
hash(transfer_from)
));
return ecrecover(digest, signature.v, signature.r, signature.s) == transfer_from.actors.signer;
}
}
Generate proper EIP-712 signatures with @ticket721/e712
in Javascript / Typescript
The @ticket721/e712
npm module can be used to properly generate signatures to make the calls.
Complete usage examples of signedTransfer
, signedApprove
, signedTransferFrom
, verifyTransfer
, verifyApprove
and verifyTransferFrom
can be found in the example implementation tests.
The module exposes a helper class, MTKNSigner
. Build it with your domain arguments and quickly generate signatures with your private key, or payloads for your web3 browser. You can also verify proofs.
import { MTKNSigner, EIP712Signature } from '@ticket721/e712';
import { Wallet } from 'ethers';
import { BN } from 'bn.js';
const domain_name = 'my mtkn';
const domain_version = '1';
const domain_chain_id = 1;
const domain_contract = '0xd0a21D06befee2C5851EbafbcB1131d35B135e87';
const transfer_recipient = '0x19C8239E04ceA1B1C0342E6da5cF3a5Ca54874e1';
const address_zero = '0x0000000000000000000000000000000000000000';
// Build helper class
const mtkn = new MTKNSigner(domain_name, domain_version, domain_chain_id, domain_contract);
// Use your own private keys
const wallet = Wallet.createRandom();
// Generate proof
const sig: EIP712Signature = await mtkn.transfer(transfer_recipient, new BN(1000), {
signer: wallet.address,
relayer: address_zero
}, {
nonce: new BN(0),
gasLimit: new BN(1000000),
gasPrice: 1000000,
reward: 500
}, wallet.privateKey) as EIP712Signature;
// Verify proofs
const verification = await mtkn.verifyTransfer(transfer_recipient, new BN(1000), {
signer: wallet.address,
relayer: address_zero
}, {
nonce: new BN(0),
gasLimit: new BN(1000000),
gasPrice: 1000000,
reward: 500
}, sig.hex);
import { MTKNSigner, EIP712Payload } from '@ticket721/e712';
import { BN } from 'bn.js';
const domain_name = 'my mtkn';
const domain_version = '1';
const domain_chain_id = 1;
const domain_contract = '0xd0a21D06befee2C5851EbafbcB1131d35B135e87';
const transfer_recipient = '0x19C8239E04ceA1B1C0342E6da5cF3a5Ca54874e1';
const address_zero = '0x0000000000000000000000000000000000000000';
const my_web3_browser_address = '0x19C8239E04ceA1B1C0342E6da5cF3a5Ca54874e1';
// Build helper class
const mtkn = new MTKNSigner(domain_name, domain_version, domain_chain_id, domain_contract);
// Generate ready-to-sign payload
const payload: EIP712Payload = await mtkn.transfer(transfer_recipient, new BN(1000), {
signer: my_web3_browser_address,
relayer: address_zero
}, {
nonce: new BN(0),
gasLimit: new BN(1000000),
gasPrice: 1000000,
reward: 500
}) as EIP712Payload;
// Sign with web3
web3.currentProvider.sendAsync({
method: 'eth_signTypedData_v3',
params: [
my_web3_browser_address,
JSON.stringify(payload)
],
from: my_web3_browser_address},
(error, result) => {
// do your stuff, signature is in result.result (if no errors)
});
- Vitalik discussing it on ethresear.ch (https://ethresear.ch/t/layer-2-gas-payment-abstraction/4513).
- Austin Thomas Griffith implementation on github (https://github.com/austintgriffith/native-meta-transactions).
- ERC-865 (#865).
- ERC-965 (#965).
- ERC-1776 Native Meta Transactions (#1776).
My goal is to make a specification that can get broadly adopted in order to start working on relay networks for mTKN
s. There is no point in developing these relays if I'm the only one agreeing with this spec. So please, help this standard come to life by giving your point of view, ideas and remarks here.
Copyright and related rights waived via CC0.