-
Notifications
You must be signed in to change notification settings - Fork 15
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
SIP-26: Non-EVM protocol support (#147)
* Add draft of SIP-26 * Remove optional header fields * Add SIP number and remove unused optional fields * Rename diagram image * Rename components diagram in document * Update diagram and add components descriptions * Rename "Address Resolution Snaps" to "Account Address Resolution Snaps" * Update components diagram * Update `resolveAccountAddress` signature * Add link to CAIP-2 and remove commented text * Add Protocol Snaps * Update components diagram * Update `resolveAccountAddress` docs * Clarify: should -> MUST * Update `resolveAccountAddress` method * Remove mention to SIP-25 * Add note about Snap events * Minor text improvements * Add mention to CAIP-2 chain IDs Co-authored-by: Gustavo Antunes <[email protected]> * Update keyring-api repo URL Co-authored-by: Charly Chevalier <[email protected]> * Add note about Accounts Router and Protocol Snaps priority * chore: update SIP-26 following latest discussions * SIP-26: Update Snaps-specific parts of the specification (#153) * SIP-26: Update Snaps-specific parts of the specification * Minor updates * Reword slightly * Update SIPS/sip-26.md Co-authored-by: Alex Donesky <[email protected]> --------- Co-authored-by: Alex Donesky <[email protected]> * chore: update title Co-authored-by: Frederik Bolding <[email protected]> * Apply suggestions from code review Co-authored-by: Alex Donesky <[email protected]> * Update SIPS/sip-26.md --------- Co-authored-by: Gustavo Antunes <[email protected]> Co-authored-by: Charly Chevalier <[email protected]> Co-authored-by: Frederik Bolding <[email protected]> Co-authored-by: Alex Donesky <[email protected]>
- Loading branch information
1 parent
f881be2
commit d2a8c29
Showing
2 changed files
with
197 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,197 @@ | ||
--- | ||
sip: 26 | ||
title: Non-EVM protocol support | ||
status: Draft | ||
author: Daniel Rocha (@danroc), Frederik Bolding (@FrederikBolding), Alex Donesky (@adonesky1) | ||
created: 2024-08-28 | ||
--- | ||
|
||
## Abstract | ||
|
||
This SIP presents an architecture to enable Snaps to expose blockchain-specific | ||
methods to dapps, extending MetaMask's functionality to support a multichain | ||
ecosystem. | ||
|
||
## Motivation | ||
|
||
Currently, MetaMask is limited to EVM-compatible networks. This proposal aims | ||
to empower developers, both first- and third-party, to use Snaps to add native | ||
support for non-EVM-compatible chains within MetaMask. | ||
|
||
## Specification | ||
|
||
> Formal specifications are written in TypeScript. | ||
### Language | ||
|
||
The key words "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT", "SHOULD", | ||
"SHOULD NOT", "RECOMMENDED", "NOT RECOMMENDED", "MAY", and "OPTIONAL" written | ||
in uppercase in this document are to be interpreted as described in [RFC | ||
2119](https://www.ietf.org/rfc/rfc2119.txt). | ||
|
||
### High-Level architecture | ||
|
||
The diagram below represents a high-level architecture of how the RPC Router | ||
integrates inside MetaMask to allow Snaps to expose protocol methods to dapps | ||
and the MetaMask clients. | ||
|
||
![High-level architecture](../assets/sip-26/components-diagram.png) | ||
|
||
- **Account Snaps**: Snaps that implement the [Keyring API][keyring-api] and are responsible | ||
for signing requests and managing accounts. | ||
|
||
- **Protocol Snaps**: Snaps that implement protocol methods that do not require | ||
an account to be executed. | ||
|
||
- **RPC Router**: Native component that forwards RPC requests to the | ||
appropriate Protocol Snap or Account Snap. | ||
|
||
- **Keyring Controller**: Native component responsible for forwarding signing | ||
requests to the appropriate keyring implementation. | ||
|
||
- **Accounts Controller**: Native component responsible for managing accounts | ||
inside MetaMask. It stores all non-sensitive account information. | ||
|
||
- **Snaps Keyring**: Native component that acts as a bridge between the | ||
Keyring Controller and the Account Snaps. | ||
|
||
### Components | ||
|
||
Here is a brief description of the components involved in this architecture | ||
which will require to be implemented or modified. | ||
|
||
#### RPC Router | ||
|
||
The RPC Router will be a new native component responsible for routing JSON-RPC | ||
requests to the appropriate Snap or keyring. | ||
|
||
To route a request, the RPC Router MUST extract the method name and [CAIP-2 chainId][caip-2] | ||
from the request object. It then determines whether the method is supported by | ||
a Protocol Snap or an Account Snap, with Account Snaps taking precedence over | ||
Protocol Snaps. | ||
|
||
If the method is supported by an Account Snap, the RPC Router forwards the | ||
request to the Keyring Controller; otherwise, it forwards the request to the | ||
appropriate Protocol Snap. | ||
|
||
#### Snaps Keyring | ||
|
||
The Snaps Keyring is an existing native component that exposes the | ||
[snap_manageAccounts][snap-manage-accs] method, allowing Account Snaps to | ||
register, remove, and update accounts. | ||
|
||
For example, this code can be used by an Account Snap to register a new account | ||
with the Account Router: | ||
|
||
```typescript | ||
// This will notify the Account Router that a new account was created, and the | ||
// Account Router will register this account as available for signing requests | ||
// using the `eth_signTypedData_v4` method. | ||
await snap.request({ | ||
method: "snap_manageAccounts", | ||
params: { | ||
method: "notify:accountCreated", | ||
params: { | ||
account: { | ||
id: "74bb3393-f267-48ee-855a-2ba575291ab0", | ||
type: "eip155:eoa", | ||
address: "0x1234567890123456789012345678901234567890", | ||
methods: ["eth_signTypedData_v4"], | ||
options: {}, | ||
}, | ||
}, | ||
}, | ||
}); | ||
``` | ||
|
||
Similar events are available to notify about the removal and update of | ||
accounts: `notify:accountRemoved` and `notify:accountUpdated`. | ||
|
||
Additionally, the Snaps Keyring expects the Account Snap to implement the | ||
Keyring API so it can forward signing requests to it through the | ||
[`keyring_submitRequest`][submit-request] method. | ||
|
||
#### Account Snaps | ||
|
||
As part of the Keyring API, non-EVM Account Snaps MUST also implement support | ||
for the `keyring_resolveAccountAddress` RPC method defined below. It is used | ||
by the RPC Router to extract the address of the account that should handle | ||
the signing request from the request object. | ||
|
||
```typescript | ||
type ResolveAccountAddressRequest = { | ||
method: "keyring_resolveAccountAddress"; | ||
params: { | ||
scope: CaipChainId; | ||
request: JsonRpcRequest; | ||
}; | ||
}; | ||
``` | ||
`scope` - The [CAIP-2][caip-2] chainId the request is targeting | ||
|
||
`request` - A `JsonRpcRequest` containing strictly JSON-serializable values. | ||
|
||
The implementation MUST return a value of the type `{ address: string }` or `null`. | ||
|
||
#### Protocol Snaps | ||
|
||
Protocol Snaps implement and expose methods that do not require an account to | ||
execute and MUST list their supported methods and notifications in their manifest file: | ||
|
||
```json5 | ||
"initialPermissions": { | ||
"endowment:protocol": { | ||
"scopes": { | ||
"<caip2_chainId>": { | ||
"methods": [ | ||
// List of supported methods | ||
], | ||
"notifications": [ | ||
// List of supported notifications | ||
] | ||
} | ||
} | ||
} | ||
} | ||
``` | ||
|
||
Additionally protocol Snaps MUST implement the `onProtocolRequest` handler: | ||
|
||
```typescript | ||
import { OnProtocolRequestHandler } from "@metamask/snap-sdk"; | ||
|
||
export const onProtocolRequest: OnProtocolRequestHandler = async ({ | ||
origin, | ||
scope, | ||
request, | ||
}) => { | ||
// Return protocol responses | ||
}; | ||
``` | ||
|
||
The interface for an `onProtocolRequest` handler function’s arguments is: | ||
|
||
```typescript | ||
interface OnProtocolRequestArguments { | ||
origin: string; | ||
scope: CaipChainId; | ||
request: JsonRpcRequest; | ||
} | ||
``` | ||
|
||
`origin` - The origin making the protocol request (i.e. a dapp). | ||
|
||
`scope` - The [CAIP-2][caip-2] chainId the request is targeting. | ||
|
||
`request` - A `JsonRpcRequest` containing strictly JSON-serializable values. | ||
|
||
Any JSON-serializable value is allowed as the return value for `onProtocolRequest`. | ||
|
||
## Copyright | ||
|
||
Copyright and related rights waived via [CC0](../LICENSE). | ||
|
||
[keyring-api]: https://github.com/MetaMask/accounts/tree/main/packages/keyring-api | ||
[snap-manage-accs]: https://docs.metamask.io/snaps/reference/snaps-api/#snap_manageaccounts | ||
[submit-request]: https://docs.metamask.io/snaps/reference/keyring-api/account-management/#keyring_submitrequest | ||
[caip-2]: https://github.com/ChainAgnostic/CAIPs/blob/main/CAIPs/caip-2.md |
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.