---
eip:
title: Gas Stations Network Protocol v2
author: Yoav Weiss <yoav@opengsn.org>, Dror Tirosh <dror@opengsn.org>, Alex Forshtat <alex@opengsn.org>
status: Draft
type: Standards Track
category: ERC
created: 2020-09-16
---
A Decentralized Meta-Transaction framework to let non-ether users to make ethereum call into the blockchain - and let a 3rd party contract pay for the gas. The caller account doesn't have to have any eth,
Incentivize nodes to run "gas stations" or "relayers" to facilitate this. Require no network changes, and minimal contract changes.
Therefore, contract owners may wish to pay for the gas to increase user acquisition, or let their users pay for gas with fiat money. Alternatively, a 3rd party may wish to subsidize the gas costs of certain contracts.
The gas stations network is an effort to solve the problem by creating an incentive for nodes to run gas stations, where gasless transactions can be "fueled up". It abstracts the implementation details from both the dapp maintainer and the user, making it easy to convert existing dapps to accept "collect-calls".
The network consists of a set of public contracts trusted by all participating dapp contracts, and a decentralized network of relay nodes (gas stations) incentivized to listen on non-ether interfaces such as web or whisper, pay for transactions and get compensated by a paymaster contract. The trusted contracts can be verified by anyone, and the system is otherwise trustless. Gas stations cannot censor transactions as long as there's at least one honest gas station. Attempts to undermine the system can be proven on-chain and offenders can be penalized.
- Increase user adoption of smart contracts by:
- Removing the user hassle of acquiring ETH. Transactions are still paid by ETH but costs can be borne by a 3rd-party contract (Paymaster). Paymasters may implement logic to verify the user's eligibility - or even charge him through other means.
- Removing the need to interact directly with the blockchain, while maintaining decentralization and censorship-resistance. Contracts can "listen" on multiple public channels, and users can interact with the contracts through common protocols that are generally permitted even in restrictive environments.
- Ethereum nodes get a revenue source without requiring mining equipment. The entire network benefits from having more nodes.
- No protocol changes required. The gas station network is self-organized via smart contracts, and dapps interact with the network by implementing an interface.
The system was built with decentralization and "mutual distrust" in its nature. We try to minimize the trust each component has on any other to limit the attack surface. In particular
- No component should be able to perform an operation on another's behalf (e.g. a client must sign each request, and a RelayRecipient contract can be assured a request was actually signed by the client)
- No component should be able to steal money from other component: e.g., relayer gets paid only if it is guaranteed it delivered a service, and paymaster must pay for transactions it approved.
- No component should be able to "grief" (cause monetarial damage) to other component.
Client
orSender
- an external account that attempt to make a call, without paying for gasRelayRecipient
- the contract that receives the relayed call from the client.Relayer
- a server that relays the request from the client to the recipient. The relayer pays for the gas, but ultimately gets reimbursed for it (with a fee) by a Paymaster.Paymaster
- a contract that pays for the actual transaction to the Relayer.RelayHub
- a singleton that mediates the calls.Forwarder
- a component to validate user's request, by checking the signature and nonce of a request.
StakeManager
- a component of the RelayHub that holds the relayers stakesPenalizer
- a component of the RelayHub that validates relayer's request - and can slash its stake if it is proven that the relayer attempted a fraud.
The relayer is built from two (or more) distinct accounts:
RelayManager
- the account the "defines" the relayer (and hold a stake)RelayWorker
- actual relayer address that sends a request. a relayer has one or more workers
Roles of the Forwarder
:
- validate the signature of the caller.
- validate the nonce in the request is unique
- call the target method on that target RelayRecipient.
- update the nonce, to prevent replay of the same request.
NOTE: the Forwarder is standalone generic contract, that can be used with different relaying frameworks - e.g multiple versions of GSN or other networks. It is further defined in EIP-xxxxx
Roles of the RelayHub
:
- Maintain a list of active relays. Senders select a
Relayer
from this list for each transaction. The selection process is discussed below. - Mediate all communication between relayers and contracts.
- Provide contracts with trusted versions of the real msg.sender and msg.data.
- Validate a relay has a stake (with a minimum value and timeout) in the StakeManager.
- Hold ETH balance of Paymasters and use them to compensate relays.
- Using the Penalizer, penalize provably-offensive relays by giving their stakes to an address providing the proof, thus keeping relays honest.
- Provide a free way for relays to know whether they'll be compensated for a future transaction.
Roles of the StakeManager
:
- hold stake for an address, with an unstake delay.
- only allow to remove the stake after the unstake delay.
- Provide the RelayHub with the stake status (value and time) for a given address.
Roles of a Relayer
node:
- For each worker address, maintain a hot wallet with a small amount of ETH, to pay for gas.
- Provide a public interface for user apps to send gasless transactions via channels such as https or whisper.
- Publish it's public interfaces and its fees (as a base fee and multiplier of the actual transaction gas cost) in
RelayHub
. - Optionally monitor reverted transactions of other relays through RelayHub, catching offending relays and claiming their stakes. This can be done by anyone, not just a relay.
Implementing a RelayRecipient
contract:
- Know the address of
Forwarder
and trust it to provide information about the transaction. - Use
_msgSender()
and_msgData()
instead ofmsg.sender
andmsg.data
, everywhere.RelayRecipient
provides these functions and gets the information fromForwarder
.
Implementing a Paymaster
-
Maintain a small balance of ETH gas prepayment deposit in
RelayHub
. Can be paid directly by theRelayRecipient
contract, or by the dapp's owner on behalf of theRelayRecipient
address. The Paymaster owner is responsible for ensuring sufficient balance for the next transactions, and can stop depositing if something goes wrong, thus limiting the potential for abuse of system bugs. In DAO usecases it will be up to the DAO logic to maintain a sufficient deposit. -
Implement a
preRelayedCall()
function to revert if it is NOT willing to accept a transaction and pay for it.preRelayedCall
is called byRelayHub
before making the actual relayed call. Some examples ofpreRelayedCall()
implementations:- Whitelist of trusted members.
- Balance sheet of registered users, maintained by the paymaster owner. Users pay the paymaster with a credit card or other non-ETH means, and are credited in the
Paymaster
balance sheet. Users can never cost the paymaster more than they were credited for. - A paymaster can provide off-chain a signed message called
approval
to a transaction sender and validate it. - Whitelist of known transactions used for onboarding new users. This allows certain anonymous calls and is subject to Sybil attacks. see below for various mitigation schemes.
-
Implement
postRelayedCall()
. This method is called after a transaction is relayed. By default, it does nothing.These two methods can be used to charge the user in paymaster-specific manner. (e.g. transfer some tokens from the user in the
preRelayedCall
, and refund the excess in thepostRelayedCall
)
GSN attempts to define the minimal trust between the various components. By "Trust" we mean expect some behaviour that cannot be otherwise enforced. Trust is achieved by auditing the code, and verifying that it can't be abused. Its easy to trust Forwarder or RelayHub code, as both are un-owned, unmodifiable contracts, so their (audited) code can be trusted by saving their addresses. When trusting an ownable (or upgradeable) contract, we trust the human owner not to do certain things (and keep his credentials safe, to prevent hackers from doing such harmful things)
- A Recipient contract trusts the forwarder contract, to forward requests only if they are signed by the sender, and are not a replay. "Trusting" means accepting the value of msgSender() as the real sender of the transaction.
The Forwarder is the only GSN component that the Recipient contract knows and trusts.
The Recipient exposes an
isTrustedForwarder
method. - A Paymaster trusts the forwarder to be "stable" (that is, return the same value when called as off-chain as a view function and on-chain transaction)
The paymaster also trusts the RelayHub to call the relayed function and finally
postRelayedCall()
afterpreRelayedCall()
returns without reverting. - The RelayHub only trusts the Penalizer and StakeManager it was initialized with.
- The sender's trust is on the RelayHub, to manage the transaction: emitting TransactionRelayed is an absolute indication that its transaction was delivered, so the client doesn't really trust other components (such the paymaster)
- The relayer trusts the paymaster to a certain level (up to "AcceptanceBudget"). Specific known paymasters can be configured by the relayer owner.
Some of the attacks on a relayed call can't be checked directly in the contract, but can be observed and proven after-the-fact. In order to prevent relayers from abusing the protocol, we provide a "penalization" mechanism in which relayers watchdog the behaviour of other relayers. In order to incentivise relayers to make these calls, they get half the stake of the penalized relayer. The other half of the stake is burnt, so that a relayer will not attempt to perform an abusive operation, and immediately "penalize" itself to redeem its entire stake.
A relayer is "penalizeable" if any of its workers submits a transaction which violates one of the rules below:
-
It fills the externalGasLimit parameter with a value that differ from the actual gasLimit of the transaction.
-
It submits 2 different transactions using the same nonce. The only difference that is allowed is raising the gas price of an already-submitted transaction.
-
A relayer attempts any transcation that is not "relayCall" to the RelayHub.
Note that this implies that when un-staking a relayer, you can only redeem the leftover eth from a worker address AFTER unstaking the relayer, since that "transfer" operation is also "penalizeable". For the same reason, an unstaked relay should never be re-staked with the same address.
The penalization check is triggered by a client sending the received transaction to a selected (or random) relayer. The client has an incentive that its own relayer will submit the correct transaction. The selected relayer has an incentive to check the transaction since it might receive the stake. Penalization can only be done to a RelayHub that is authorized on the StakeManager, or during "unstakeDelay" after unauthorizeHub.
- Relayer starts listening as a web app (or on some other communication channel).
- If starting for the first time (no key yet), generate a key pair for Relayer's manager and worker addresses.
- The Relayer's owner should fund the manager address with some eth.
- Relayer's owner sends the required stake to
StakeManager
by callingStakeManager.stakeForAddress(address relay, uint unstakeDelay)
. - Relayer's owner calls
StakeManager.authorizeHubByOwner(relayManager,relayHub)
- Relay calls
RelayHub.registerRelayServer(baseFee, pctFee, url)
with the relayer's configuredtransaction fee
(as a base fee and a multiplier on transaction gas cost), and a URL for incoming transactions. RelayHub
ensures that Relayer has an authorization and sufficient stake in theStakeManager
.RelayHub
emits an event,RelayAdded(Relay, owner, transactionFee, relayStake, unstakeDelay, url)
.Relayer
goes to sleep and waits for signing requests.- The Relayer should update its minGasPrice, using some "gas oracle" This is the minimum gas price the relayer accepts from clients, to make sure the get mined in reasonable time.
- Still, the Relayer should check periodically for pending transactions (due to gas fluctuation, it can happen that even the above minGasPrice was too low, and takes too long to mine) The relayer is allowed to modify (raise) the gas price on the transaction: it will not get compensated for this higher gas price, but at least the tx won't be blocked, and the worker will be released to process more requests.
- Relayer's owner (the address that initially funded it) calls
StakeManager.unlockStake(relayManager)
. - The StakeManager validates its the actual owner, and it does have a stake.
- From that point, no transaction can go through this relayer. At this point the relayer can still be penalized for past offenses.
- Once the owner's unstake delay is over, owner calls
RelayHub.unstake()
, and withdraws the stake. - The relayer should stop the
Relay
shuts down.
- The same RelayManager (but not relay workers!) can be used by multiple Hubs (e.g, when switching to a new hub version, and relay owner would like to provide service through both the old hub and new hub, without requiring to add new stake.
- the Relayer is brought up with the new RelayHub (with the registering process above, but without re-staking)
- When the old RelayHub is no longer in use, the owner should call
StakeManager.unauthorizeHubByOwner(relayManager,relayHub)
- The StakeManager will remove the authorization, and the relayManager will no longer be able to process requests by this RelayHub.
- When receiving the "HubUnauthorized" events, the Relayer will start a countdown of the unstake-delay.
- Only after this countdown is over, it will transfer the eth balance of the relay workers to the manager (before that time, the "transfer eth" operation balance can cause the relayer to be penalizeable)
- The client lookup mechanism filters out relayers for which a "HubUnauthorized" event was sent
-
Sender
selects a liveRelay
from RelayHub's list by looking at events fromRelayHub
in the pastlookup window
. Each relayer address is saved once (that is, a relayer is considred "active" regardless of how many requests it had processed within the lookup window).Sender
sends a "ping" request to the selectedRelay
.- This ping returns the relayer manager, worker, minGasPrice.
-
The sender may maintain a list of "preferred relayers". These are always moved to the beginning of the list and tried first, before other relayers (i.e probably dapp-owned relayers).
-
The sender may sort relayers based on its own criteria. Selection may be based on a mix of:
- Relayer published transaction fees.
- Relayer stake size and lock-up time.
- Relayer minGasPrice (filter out relayers that require higher gasprice).
- Recent relayer transactions (visible through
TransactionRelayed
events fromRelayHub
). - Optionally, reputation/blacklist/whitelist held by the sender app itself, or its backend, on per-app basis (not part of the gas stations network).
-
The Default sorting algorithm is:
- Preferred relays are always first, in their order from the configuration file.
- Sort relayers by calculated fee (for this transaction) .
- Filter out relayers with high "minGasPrice".
- Downscore (move to the end of the list) relays that previously failed this client (see below).
-
To select a relayer to use from the list, the client:
- Picks the next 3 relayers (without mixing "preferred" with normal relayers).
- Sends a "ping" request to all of them.
- Verifies the RelayManager and RelayHub in that ping response.
- Saves the RelayWorker address (used below).
- Takes the first relayer to answer the ping.
-
Sender prepares the transaction with Sender's address, the recipient address, the actual transaction data, Relay's transaction fee, gas price, gas limit, its current nonce from
Forwarder.nonces
, RelayHub's address, and Relay-worker's address, and then signs it. -
The Sender makes a view-call to
RelayHub.relayCall()
, to make sure the call is valid, and paymaster agrees to pay. -
The Sender may also make the explicit verifications:
RelayHub.balances[forwarder]
holds enough ETH to pay Relay's fee.RelayWorkder.balance
has enough eth to send the transaction.
-
Sender reads the Relay worker's current
nonce
value and decides on themax_nonce
parameter. -
Sender sends the signed transaction and metadata to Relayer's web interface.
-
Relayer
receives the request, and verify the client's request:- The required gasPrice is acceptable by the relayer (too-low gas price might stall this worker for some time).
- The next nonce to use is within "nonce_gap" from the nonce of the last mined transaction of this worker.
-
Relayer
receives the request, and make a view call toRelayHub.relayCall()
, to verify it will get compensated: -
The
RelayHub.relayCall
validates (and reverts otherwise) that:- The relay manager is registered and staked.
- The sender of the request is indeed a relayWorker of that manager.
- The paymaster can pay for the max possible price.
- The relayhub provided enough gas to complete the request.
- The paymaster's preRelayedCall accepted the call (and didn't revert).
-
If any of Relayer's checks fail, it returns an error to sender, and doesn't proceed.
-
Relayer submits the signed relayCall transaction to the blockchain.
-
Relayer immediately returns the signed relayCall transaction to the sender. This step is discussed below, in attacks/mitigations.
-
Sender
receives the wrapped transaction and verifies that:- It's a valid call to
RelayHub.relayCall
, with its own transaction, from RelayWorker's address. - The transaction's ethereum nonce is within "nonce_gap" from RelayWorker's current nonce.
- The
RelayWorker
is sufficiently funded to pay for it.
- It's a valid call to
-
If any of sender's checks fails, it goes back to selecting the next relayer.
- As discussed above, this failed relayer is now downscore'd by this sender, so that it will avoid to select it again in future transaction.
-
Sender
may also submit the raw wrapped transaction to the blockchain without paying for gas, through any Ethereum node. This submission is likely ignored because an identical transaction is already in the network's pending transactions, but no harm in putting it twice, to ensure that it happens. This step is not strictly necessary, for reasons discussed below in attacks/mitigations, but may speed things up. -
Sender
monitors the blockchain, waiting for the transaction to be mined. The transaction was verified, with Relayer's current nonce, so mining must be successful unless Relayer submitted another (different) transaction with the same nonce. -
If mining fails due to such attack, sender may call
RelayHub.penalizeRepeatedNonce
through another relay, to collect his reward and burn the remainder of the offending relay's stake, and then go back to selecting a new Relayer for the transaction. See discussion in the attacks/mitigations section below. -
RelayHub
receives the transaction:- Verifies the transaction is sent from a worker of a registered relay, and that's the same worker signed in the transaction.
- Verifies the tx.gasPrice is at least as high as the request gasPrice.
- Verifies the given acceptanceBudget is above paymaster's acceptanceBudget (see #AcceptanceBudget section).
- Verifies the transaction is given enough gas to cover all gas limits (preRelayedCall, actual call and postRelayedCall).
- Verifies the paymaster can pay for the max possible price of the transaction.
- Makes a call to
paymaster.preRelayedCall
, to make sure it accepts the request.- The paymaster makes sure it trusts the forwarder of the request.
- The paymaster then performs its other validations.
- In case the paymaster reverts, a "TransactionRejectedByPaymaster" is emitted.
- The RelayHub sends the transaction to the recipient through the forwarder.
The forwarder verifies the nonce and signature, and then call the target recipient.
When passing gas to
Forwrader.execute()
, enough gas is preserved byRelayHub
, for post-call handling. Recipient may run out of gas, butRelayHub
never does. - The
Forwarder
verifies the signature of the sender, and that the nonce is correct, and increment the nonce. - The
Forwarder
makes a call to the recipient, after appending the sender's address at the end of the msg.data. This way, the recipient can safely assume that when it receives a call through the Forwarder, themsg.sender
is passed as the last 20 bytes of the msg.data. - Recipient contract handles the transaction.
RelayHub
calls paymaster'spostRelayedCall
.RelayHub
checks call's return value of call, and emitsTransactionRelayed(address relay, address from, address to, bytes4 selector, uint256 status, uint256 chargeOrCanRelayStatus)
.RelayHub
transfers ETH balance from paymaster toRelayManager
, to pay the transaction fee, based on the measured transaction cost.
-
Relayer
keeps track of sent transactions. If the transaction doesn't get mined, it should boost it by putting a higher gas price. Any other modification to the transaction is not allowed, and is considered a violation of the protocol, for which the relayer will get penalized. Note that without boosting it, the relayWorker is blocked, and can't send other transactions.
The GSN protocol is based on an assumption of the client (and then of the relayer) that running a transaction in view mode will get the same result as running it on-chain.
The relayer trusts that if the view call succeeds, its OK to put it on-chain and it will be refunded (by the paymaster) for its cost (+fee).
However, it is possible to write contracts (paymaster or recipient) that will cause a transaction to be "unstable", and thus succeed off-chain and later fail on-chain. In such a case, the relayer will suffer a loss of funds.
The RelayHub emits TransactionRejectedByPaymasterEvent
event in this case - but it can't tell whether it was because of the client,
the paymaster or the relayer itself.
That is, a relayer can tell it wasn't itself to cause it - but it can't say so on events emitted by other relayers.
The relayers need a way to detect and mitigate such cases to keep profitable.
Examples of such "unstable" cases (but definitely not the only cases):
-
require(block.number % 1 == 0)
- this naive code will "randomly" fail. -
Parallel sending the same transaction to multiple relays at the same time: obviously, one will succeed, and the other swill fail (since the sender's nonce in the forwarder was modified by the first call) . This attack can be performed by a client without direct help of a paymaster (that is, a client attacks the paymaster and relayer).
A paymaster can mitigate it by using off-chain mechanism to blacklist offending clients (see "Off-chain verification" below).
Note that in either case, the attack is not "free": in the first case, there is a chance that it gets through anyway. In the latter case, one transaction gets paid, so the attack "cost" is higher than the actual damage to each relayer (though total attack on all relayers can be higher).
The above "parallel sending" can happen in legitimate cases (e.g. a client sends a request but fails to receive a response, so it re-sends the request through another relayer) It is assumed that such cases are rare, so a mitigation mechanism should be able to tolerate such cases.
The mitigation mechanisms below take into account the fact that such "unstable" transaction might occur without any real attack on the network - though they should be rare (e.g. a client connection to a relayer times out, and it re-send the request through another relayer. It is possible that both requests will be put on chain, and thus one might see it as an "attack")
The GSN protocol contains several mechanisms to mitigate such cases:
- AcceptanceBudget: For any given transaction, the Relayer risks at most "acceptanceBudget" of gas, and it is guaranteed to be refunded for paymaster rejects with higher gas usage. The Paymaster advertises (through Paymaster.getLimits()) its acceptanceBudget. The default of 150k is enough for most but extreme cases (such as zero-knowledge-based mixers).
Note that by "paymaster reject" we count both preRelayedCall's gas usage AND forwarder's check for signature and nonce. Since forwarder's checks take ~30kgas, the default preRelayedCallGasLimit was set to 100k, so that the sum of both are below the 150kgas acceptanceBudget (and will stay so even if opcode gas usage will be modified).
Note that if the Relayer's owner knows the Paymaster code and trust it to be "stable" (which also means the trusted Forwarder is "stable"), then it can allow for higher acceptanceBudget. This is done by adding the paymaster's address to the "trustedPaymasters" list in the relayer's configuration.
- AlertedMode: this is a mechanism to help the relayer mitigate the 2nd unstable case above:
When the relayer detects an event "TransactionRejectedByPaymasterEvent", it knows it is probably because of parallel sending of such transaction. In alerted mode, the relayer waits a random time before sending the transaction. This way, there is a chance that the other relayer that was given this transaction will send it first, and this relayer will be able to see it (on-chain or in the mempool), and avoid re-sending the same TX.
The relayer will exit "alerted" mode after a preconfigured time without any rejected transaction.
This does not entirely eliminate the chance of parallel transactions, but reduces it, hence makes the attack more expensive (the transaction gets paid, and "griefs" fewer relayers).
- Paymaster Reputation: This is a stronger mechanism that lets a relayer learn over time which paymasters are trusted.
- A new paymaster is given a reputation of "1"
- For each relayed transaction the relayer increases the reputation of a paymaster.
- For each failed transaction, the relayer reduces the reputation of the paymaster.
- Reputation<5 throttles requests at 1 per minute (i.e, no new tx until previous one confirmed on chain).
- Reputation is maxed at 50.
- If reputation drops 20% within a time-window (e.g. 1 hour), the paymaster is blocked for 1 day, and reputation is dropped to zero.
- If reputation drops below -2 (=3 failures), the paymaster is blocked for 1 day - or even completely.
A Paymaster is the on-chain entity that verifies the request and ultimately pays for the transaction. One important aspect of the paymaster is that it has to block greifing attempts.
We see several types of Paymasters:
-
Test-Only EverythingAccepted - the most naive paymaster, which simply accept requests. It can only be used on testnets, as it will be immediately grief by wily hackers.
-
Whitelisting - In case the senders are known, the paymaster can whitelist the sender addresses it accepts. This is good if you already have an on-chain repository of known addresses (e.g, a DAO may pay for voting by the DAO participants).
-
Off-chain verification - in many cases, an off-chain mechanism for user verification can be used, such as OAuth, Email, SMS, Captcha, etc. In this case, the off-chain service will perform a verification, and sign the request. The on-chain Paymaster will receive that signature as
approvalData
. The only thing the on-chain paymaster needs to know, is what signer addresses it supports. -
Off-chain payment - a special case of the off-chain verification is "pay for gas": use a payment service (such as stripe or paypal), and sign the request, to be checked by the paymaster.
-
Pay-with-token - a paymaster that manages the balance of a user, and deducts the transation cost. While the user doesn't need to have ETH, he does need to have the right token, which means that the user needs to set an approval for the paymaster to withdraw from it.
The rationale for the gas stations network design is a combination of two sets of requirements: Easy adoption, and robustness.
For easy adoption, the design goals are:
- No network changes.
- Minimal changes to contracts, apps and frameworks.
The robustness requirement translates to decentralization and attack resistance. The gas stations network is decentralized, and we have to assume that any entity may attack other entities in the system.
Specifically we've considered the following types of attacks:
- Denial-of-service attacks against individual senders, i.e. transactions censorship.
- Denial-of-service and financial attacks against individual relays.
- Denial-of-service and financial attacks against individual contracts (paymasters).
- Denial-of-service attacks against the entire network, either by attacking existing entities, or by introducing any number of malicious entities.
GSN attempts to be as gas-neutral as possible. That is, a relayer should be compensated for the actual gas used (it can ask for fee, but should not
get over-compensated).
Likewise, a paymaster should be able to estimate the gas to be used so it can perform its own accounting (e.g, charge the user with equivalent tokens).
In Ethereum method call, a method is given a gas limit, and this gas is used first to make the call (copy params, etc) and then for actual execution.
However, the gas limit itself is not available for the contract code itself.
So with GSN, we require the relayer to pass externalGasLimit
parameter, which should be identical to the gas it has given the relayCall()
.
(This value can't be validated during the transaction itself, but putting a wrong value makes the relayer "penalizeable")
at any given point during a method call, calling externalGasLimit-gasleft()
is the total gas usage up to this point, including the cost of making the call (for public function, this includes the cost of putting the transaction on-chain (4gas for each zero byte, 16 for nonzero byte))
the relayCall calculate the total cost this way, just before refunding the relayer and emitting the final TransactionRelayed.
Of course, these instructions themselves are outside the gas calculation, so we have "gasOverhead", which is calculated by comparing the estimation to actual gasUsed.
For the Paymaster, we want to calculate the gasUseWithoutPost (its the paymaster's developer job to estimate the gas usasge of the postRelayedCall itself). This method is called within an innerRelayedCall, so we had to make the same structure: pass the "externalGasLimit"
In order to support high throughput through GSN, we have 2 mechanisms:
-
nonceGap: Clients are aware that its possible that the relayer will relay more than one transaction in a block. That is, even if the last mined relay worker's nonce is "N", the client's transaction might get nonce of "N+3", meaning there are extra 2 transactions not yet mined. The client specifies what is the largest "gap" it allows. The relayer should reject the request if it is unable to fulfil the client's request of smaller gap than current nonce. If the relayer signs the transaction with a higher nonce, it is not penalizeable, but still, the client will move to another relayer to send the TX.
Since on that 2nd relayer it might get mined faster, the first relayer will have its transaction reverted. While it could "blame" the client, it can't do much about it, since this client will probably won't send any more transactions through it.
The relayer should provide an API to the client to return all pending transactions since nonce "N". Using this API, the client can validate that its own transaction will get mined in due time, by validating these intermediate transactions:
- They are all properly signed by the relay worker.
- They are all in the mempool (the client may put them into the mempool by itself).
- The worker has enough gas to submit them all (including the client's transaction)
- Their gasPrice is not below the client's gas.
-
Multiple workers: A Single relayer may have multiple worker addresses. The RelayHub is configured with the highest number of workers (currently 10). Each worker is independent with its nonce (and gas price) management, but they share the same manager stake. Multiple workers reduce the need for large "nonceGap", but can't completely eliminate it. That is, a client should expect that sometimes it will receive its signed transaction using a nonce that is higher than current worker nonce, and use the "nonceGap" mechanism above, to validate any intermediate transaction.
For GSN2, we opted to use EIP-712 as a method to show the transaction that gets signed to the user, before sending it to the server. We believe that signing an opaque hex blob is undesired. While the current EIP-712 view dialogs don't show much information, we anticipate that in the future, wallet applications will be able to show better view for meta-transactions, just like they recognize some normal transactions, such as token transfers and DeFi operations. Actually, the ForwardRequest matches the 6 fields of an Ethereum transaction - from,to,value,gas,nonce,data To these, GSN adds the GSN-specific fields, like relaying fee and paymaster.
Attack: Relay attempts to censor a transaction by not signing it, or otherwise ignoring a user request.
Relay is expected to return the signed transaction to the sender, immediately. Sender doesn't need to wait for the transaction to be mined, and knows immediately whether it's request has been served. If a relay doesn't return a signed transaction within a few seconds, sender cancels the operation, drops the connection, and switches to another relay. It also marks Relay as unresponsive in its private storage to avoid using it in the near future.
Therefore, the maximal damage a relay can cause with such attack, is a one-time delay of a few seconds. After a while, senders will avoid it altogether.
Attack: Relay attempts to censor a transaction by signing it, returning it to the sender, but never putting it on the blockchain.
This attack will backfire and not censor the transaction. The sender can submit the transaction signed by Relay to the blockchain as a raw transaction through any node, so the transaction does happen, but Relay may be unaware and therefore be stuck with a bad nonce which will break its next transaction.
Attack: Relay attempts to censor a transaction by signing it, but publishing a different transaction with the same nonce.
Reusing the nonce is the only DoS performed by a Relay, that cannot be detected within a few seconds during the http request.
It will only be detected when the malicious transaction with the same nonce gets mined and triggers the RelayHub.TransactionRelayed
event.
However, the attack will backfire and cost Relay its entire stake.
Sender has a signed transaction from Relay with nonce N, and also gets a mined transaction from the blockchain with nonce N, also signed by Relay.
This proves that Relay performed a DoS attack against the sender.
The sender calls Penalizer.penalizeRepeatedNonce(bytes transaction1, bytes transaction2)
, which verifies the attack, confiscates Relay's stake,
and sends half of it to the sender who delivered the penalizeRepeatedNonce
call. The other half of the stake is burned by sending it to address(0)
. Burning is done to prevent cheating relays from effectively penalizing themselves and getting away without any loss.
The sender then proceeds to select a new relay and send the original transaction.
The result of such attack is a delay of a few blocks in sending the transaction (until the attack is detected) but the relay gets removed and loses its entire stake. Scaling such attack would be prohibitively expensive, and actually quite profitable for senders and honest relays.
Attack: Relay attempts to censor a transaction by signing it, but using a nonce higher than it's current nonce.
In this attack, the Relay did create and return a perfectly valid transaction, but it will not be mined until this Relay fills the gap in the nonce with 'missing' transactions.
This may delay the relaying of some transactions indefinitely. In order to mitigate that, the sender includes a max_nonce
parameter with it's signing request.
It is suggested to be higher by 2-3 from current nonce, to allow the relay process several transactions.
When the sender receives a transaction signed by a Relay he validates that the nonce used is valid, and if it is not, the client will ignore the given relay and use other relays to relay given transaction. Therefore, there will be no actual delay introduced by such attack.
Attack: Paymaster attempts to burn relays funds by implementing an inconsistent preRelayedCall() and using multiple sender addresses to generate expensive transactions, thus performing a DoS attack on relays and reducing their profitability.
In this attack, a contract sets an inconsistent preRelayedCall (e.g. revert on even blocks), and uses it to exhaust relay resources through unpaid transactions. Relays can easily detect it after the fact. If a transaction goes unpaid, the relay knows that the Paymaster contract's preRelayedCall has acted inconsistently, because the relay has verified its view function before sending the transaction. It might be the result of a rare race condition where the contract's state has changed between the view call and the transaction, but if it happens too frequently, relays will blacklist this paymaster and refuse to serve transactions to it. Each offending paymaster can only cause a small damage (e.g. the cost of 2-3 transactions) to a relay, before getting blacklisted.
Relays may also look at paymaster's history on the blockchain, looking for past unpaid transactions (reverted by RelayHub without pay), and denying service to contracts with a high failure rate. If a contract caused this minor loss to a few relays, all relays will stop serving it, so it can't cause further damage.
This attack doesn't scale because the cost of creating a malicious contract is in the same order of magnitude as the damage it can cause to the network. Causing enough damage to exhaust the resources of all relays, would be prohibitively expensive.
The attack can be made even more impractical by setting RelayHub to require a stake from paymasters before they can be served, and enforcing an unstaking delay, so that attackers will have to raise a vast amount of ETH in order to simultaneously create enough malicious contracts and attack relays.
This protection is probably an overkill, since the attack doesn't scale regardless.
Attack: User attempts to rob dapps by registering its own relay and sending expensive transactions to dapps.
If a malicious sender repeatedly abuses a paymaster by sending meaningless/reverted transactions and causing the paymaster to pay a relay for nothing, it is the paymaster's responsibility to blacklist that sender and have its preRelayedCall function revert for that sender. Collect calls are generally not meant for anonymous senders unknown to the recipient. Dapps that utilize the gas station networks should have a way to blacklist malicious users in their system and prevent Sybil attacks.
A simple method that mitigates such Sybil attack, is that the dapp lets users buy credit with a credit card, and credit their account in the dapp contract, so preRelayedCall() only succeeds for users that have enough credit, and deduct the amount paid to the relay from the user's balance, whenever a transaction is relayed for the user. With this method, the attacker can only burn its own resources, not the dapp's.
A variation of this method, for free dapps (that don't charge the user, and prefer to pay for their users transactions) is to require a captcha during user creation in their web interface, or to login with a Google/Facebook account, which limits the rate of the attack to the attacker's ability to open many Google/Facebook accounts. Only a user that passed that process is given credit in RelayRecipient. The rate of such Sybil attack would be too low to cause any real damage.
Registering a relay requires placing a stake in RelayHub, and the stake can only be withdrawn after the relay is unregistered and a long cooldown period has passed, e.g. a month.
Each unreliable relay can only cause a couple of seconds delay to senders, once, and then it gets blacklisted by them, as described in the first attack above. After it caused this minor delay and got blacklisted, the attacker must wait a month before reusing the funds to launch another unreliable relay. Simultaneously bringing up a number of unreliable relays, large enough to cause a noticeable network delay, would be prohibitively expensive due to the required stake, and even then, all those relays will get blacklisted within a short time.
Transactions include a nonce. RelayHub maintains a nonce (counter) for each sender. Transactions with bad nonces get reverted by RelayHub. Each transaction can only be relayed once.
Attack: User does not execute the raw transaction received from the Relayer, therefore blocking the execution of all further transactions signed by this relayer
The user doesn't really have to execute the raw transaction. It's enough that the user can. The relationship between relay and sender is mutual distrust. The process described above incentivizes the relay to execute the transaction, so the user doesn't need to wait for actual mining to know that the transaction has been executed.
Once relay returns the signed transaction, which should happen immediately, the relay is incentivized to also execute it on chain, so that it can advance its nonce and serve the next transaction. The user can (but doesn't have to) also execute the transaction. To understand why the attack isn't viable, consider the four possible scenarios after the signed transaction was returned to the sender:
- Relay executes the transaction, and the user doesn't. In this scenario the transaction is executed, so no problem. This is the case described in this attack.
- Relay doesn't execute the transaction, but the user does. Similarly to 1, the transaction is executed, so no problem.
- Both of them execute the transaction. The transactions are identical in the pending transactions pool, so the transaction gets executed once. No problem.
- None of them execute the transaction. In this case the transaction doesn't get executed, but the relay is stuck. It can't serve the next transaction with the next nonce, because its nonce hasn't been advanced on-chain. It also can't serve the next transaction with the current nonce, as this can be proven by the user, having two different transactions signed by the same relay, with the same nonce. The user could use this to take the relay's nonce. So the relay is stuck unless it executes the transaction.
As this matrix shows, the relay is always incentivized to execute the transaction, once it returned it to the user, in order to end up in #1 or #3, and avoid the risk of #4. It's just a way to commit the relay to do its work, without requiring the user to wait for on-chain confirmation.
The gas stations network is implemented as smart contracts and external entities, and does not require any network changes.
Dapps adding gas station network support remain backwards compatible with their existing apps/users. The added methods apply on top of the existing ones, so no changes are required for existing apps.
A working implementation of the gas stations network is being developed by OpenGsn.
Copyright and related rights waived via CC0.