Hi all, at MetaMask we’ve been working on a new feature proposal for web3 wallets, aimed at increasing usability while maintaining coherent security. You can read about our motivations in more detail in this blog post.
The proposal below is a syndication of this EIP branch submitted as this EIP issue.
eip: TBD
title: Wallet Permissions System
author: Dan Finlay (@danfinlay), Erik Marks (@rekmarks)
discussions-to: TBD
status: Work in progress (WIP)
type: Standard Track
category: Interface
created: 2019-08-22
requires: 1474
Simple Summary
A proposed standard interface for restricting and permitting access to security-sensitive methods within a restricted web3 context like a website or “dapp”.
Abstract
Web3 JavaScript wallet browsers may implement wallet_getPermissions
and wallet_requestPermissions
. This provides a standard interface for requesting permissions and checking a domain’s current permissions status.
Motivation
Web3 Wallets are built around the responsibility of mediating the interactions between untrusted applications and a user’s keys on their computer, getting appropriate consent from the user.
Today web3 browsers like MetaMask always prompt on a per-action basis. This provides security at the cost of substantial user friction. We believe that a single permissions request can achieve the same level of security with vastly improved UX.
The pattern of permissions requests is common around the web, from login with Facebook, Twitter, Github, and even Apple, making it a very familiar pattern.
Many web3 applications today begin their sessions with a series of repetitive requests:
- Reveal your wallet address to this site.
- Switch to a preferred network.
- Sign a cryptographic challenge.
- Grant a token allowance to our contract.
- Send a transaction to our contract.
Many of these (and possibly all), and many more (like decryption), could be generalized into a set of human-readable permissions prompts on the original sign-in screen, and additional permissions could be requested only as needed.
On the user’s end, each of these permissions could be individually rejected (unchecked), or even attenuated, or adjusted to meet the user’s terms (for example, a sign-in request could have a user-added expiration date, and a token allowance could be adjusted by the user when it is requested), making the web3 login a sort of user-revisable terms of use.
Specification
This proposal adds two new methods to a wallet’s web3 provider API:
wallet_getPermissions
wallet_requestPermissions
The wallet_getPermissions
method is used for getting an array of current permissions (empty by default), while the wallet_requestPermissions
method is used for an application to request additional permissions.
These two methods are used to restrict a few hypothetical “restricted methods”. The first such method we would suggest should be included as part of the standard is eth_accounts
.
In this framework, the permission for a user to reveal their accounts would look like this:
const response = await provider.send({
method: 'wallet_requestPermissions',
params: [{
'eth_accounts': {},
}]
})
If this request was rejected, it would throw an error with a code
value equal to 4001
, per EIP 1193 errors, which the MetaMask team has canonized in a module eth-json-rpc-errors.
If the request is accepted by the user, then subsequent requests to eth_accounts
will succeed, and return an accounts array as usual.
A call to wallet_getPermissions
will then return a permissions schema object that describes the current permission.
const response = await provider.send({
method: 'wallet_getPermissions'
})
Would return a value something like this:
[
{
parentCapability: 'eth_accounts',
caveats: [
{
type: 'filterResponse',
value: ["0x0c54fccd2e384b4bb6f2e405bf5cbc15a017aafb"]
}
]
}
]
The term parentCapability
comes from the ocap-ld spec, which these permissions objects are based on.
You can see above how internally the user-selected account is transformed into a caveat
, which is a restriction on the response values, in this case ensuring the page can only be notified of approved accounts. This also means this permissions system is forward-extensible to support logging into a page with multiple accounts.
Rationale
While the current model of getting user consent on a per-action basis has high security, there are huge usability gains to be had bo getting more general user consent which can cover broad categories of usage, which can be expressed in a more human-readable way. This pattern has a variety of benefits to offer different functions within a web3 wallet.
The eth_sendTransaction
method itself could be a restricted method (requested by default with the provider.enable()
method), and the user could at sign-in time decide whether they wanted to require confirmations, approve all transactions, or only approve transactions to a certain contract, or up to a certain token limit, for example. By restricting this method by default, wallets could prevent sites from spamming the user with popups.
If eth_call
were a restricted method, then random websites would not be able to drain a user’s subscription to a hosted provider, making it easier to protect services like Infura against DDoS attacks.
On-chain actions could be represented as a permission under this model, for example, the permission to send an allowance-setting transaction to a specific token address is virtually equialent to the approval of that transaction, except the site could choose to only invoke the transaction when it was needed. This could allow a standard interface for applications to request permissions which may require different actions depending on different types of accounts (hot wallets, hardware wallets, cold wallets, contract accounts).
The requestPermissions
method could be expanded to include other options related to the requested permissions, for example, sites could request accounts with specific abilities. For example, a website like an exchange that requires signTypedData_v3
(which is not supported by some hardware wallets), might want to specify that requirement, maybe like this:
provider.send({
method: 'requestPermissions',
params: [
{
'eth_accounts': {
requiredMethods: ['signTypedData_v3']
}
}
]
})
That type of API will also be up for discussion on The MetaMask repository.
This would allow the wallet to limit the user’s options to valid ones, and allows dapps to ensure selected accounts are compatible with their service, while preserving the user’s privacy regarding how they are storing their keys.
Implementation
We have a branch of MetaMask available now which adds these methods via an rpc-engine middleware called json-rpc-capabilities-middleware (or often RpcCap
internally, for short).
The latest build of this branch of MetaMask can be downloaded from the draft pull request (look for the latest post by @MetaMaskBot
). A guide to adding a custom build of MetaMask to Chrome can be found here.
This branch of MetaMask can be used with this sample site (source), which uses a couple sample permissions for demonstration purposes:
-
readYourProfile
: We have bundled this build with an imaginary concept of a local “profile”, a simple POJO. Eventually this could be extended to instead expose the user’s 3box profile. -
writeToYourProfile
: This permission allows the requesting app to freely update/edit the user’s profile. -
sendEther
: A permission allowing the sending of transactions.
It is notable that this branch is the first version of MetaMask that allows you to be connected to each site with a different account, which persists on that site, along with any other permissions granted to the site.
You can get more detailed API and type information on the RpcCap repository’s readme.
New hypothetical and proposed permissions can be easily added to the restrictedMethods
hash in the MetaMask permissions controller.
Copyright
Copyright and related rights waived via CC0.