Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add signature insights documentation #1228

Merged
merged 9 commits into from
Mar 25, 2024
Merged
Show file tree
Hide file tree
Changes from 8 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Binary file added snaps/assets/features/signature-insights.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added snaps/assets/signature-insights-permission.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added snaps/assets/signature-insights-warning.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
2 changes: 1 addition & 1 deletion snaps/features/custom-name-resolution.md
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ For example, to resolve Ethereum Mainnet domains, add the following to your Snap

### 2. Implement custom name resolution

Export an [`onNameLookup`](../reference/entry-points.md#onnamelookup) entry point, which receives a
Expose an [`onNameLookup`](../reference/entry-points.md#onnamelookup) entry point, which receives a
`chainId` and either a `domain` or an `address`.
The following example implements a very basic resolution from Unstoppable Domains domain names to
Ethereum addresses in `onNameLookup`:
Expand Down
170 changes: 170 additions & 0 deletions snaps/features/signature-insights.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,170 @@
---
description: Provide insights to your users in MetaMask's signature confirmation flow.
sidebar_position: 7
sidebar_custom_props:
flask_only: true
---

import Tabs from '@theme/Tabs';
import TabItem from '@theme/TabItem';

# Signature insights

:::flaskOnly
:::

You can provide signature insights before a user signs a message.
For example, you can warn the user about potentially dangerous signature requests.

## Steps

### 1. Request permission to display signature insights

Request the [`endowment:signature-insight`](../reference/permissions.md#endowmentsignature-insight)
permission by adding the following to your Snap's manifest file:

```json title="snap.manifest.json"
{
"initialPermissions": {
"endowment:signature-insight": {}
}
}
```

If you need to receive the origin of the signature request, add `allowSignatureOrigin` to the
permission object, and set it to `true`:

```json title="snap.manifest.json"
{
"initialPermissions": {
"endowment:signature-insight": {
"allowSignatureOrigin": true
}
}
}
```

When requesting this permission, the following displays in the MetaMask UI when a user installs the Snap:

<p align="center">
<img src={require("../assets/signature-insights-permission.png").default} alt="Signature insights permission" style={{border: "1px solid #DCDCDC"}} />
</p>

### 2. Implement `onSignature` and export it from `index.ts`

Expose an [`onSignature`](../reference/entry-points.md#onsignature) entry point, which receives a
`signature` object.
The shape of this object depends on the chain and the signing method used.
This is why it's typed as `Record<string, unknown>`.

For Ethereum and Ethereum-compatible chains, the `signature` object can have one of the following
shapes, depending on the signing method used:

<Tabs>
<TabItem value="eth_sign">

```typescript
interface EthSignature {
from: string;
data: string;
signatureMethod: "eth_sign";
}
```

</TabItem>
<TabItem value="personal_sign">

```typescript
interface PersonalSignature {
from: string;
data: string;
signatureMethod: "personal_sign";
}
```

</TabItem>
<TabItem value="eth_signTypedData">

```typescript
interface SignTypedDataSignature {
from: string;
data: Record<string, any>[];
signatureMethod: "eth_signTypedData";
}
```

</TabItem>
<TabItem value="eth_signTypedData_v3">

```typescript
interface SignTypedDataV3Signature {
from: string;
data: Record<string, any>;
signatureMethod: "eth_signTypedData_v3";
}
```

</TabItem>
<TabItem value="eth_signTypedData_v4">

```typescript
interface SignTypedDataV4Signature {
from: string;
data: Record<string, any>;
signatureMethod: "eth_signTypedData_v4";
}
```

</TabItem>
</Tabs>

Your Snap should use `signatureMethod` as the source of truth to identify the signature scheme it is
providing insights for.

Once you've identified the signature object, your Snap can run any logic, including calling APIs.
Then, your Snap must either return `null` if it has no insights to provide, or an object with a
`content` property and an optional `severity` property as specified in the
[`onSignature`](../reference/entry-points.md#onsignature) entry point.

:::caution
Due to current MetaMask UI limitations, signature insights will only be displayed if your Snap's
logic deems the signature to be one that a user shouldn't sign, that is, if you return a severity
level of `SeverityLevel.Critical`.
:::

The following is an example implementation of `onSignature`:

```typescript title="index.ts"
import type { OnSignatureHandler, SeverityLevel } from "@metamask/snaps-sdk";
import { panel, heading, text } from "@metamask/snaps-sdk";

export const onSignature: OnSignatureHandler = async ({
signature,
signatureOrigin,
}) => {
const insights = /* Get insights based on custom logic */;
return {
content: panel([
heading("My Signature Insights"),
text("Here are the insights:"),
...(insights.map((insight) => text(insight.value)))
]),
severity: SeverityLevel.Critical
};
};
```

When your Snap returns a signature insight with a `severity` of `SeverityLevel.Critical`, the custom
UI displays in a modal after the user selects the **Sign** button.
For example:

<p align="center">

![Signature insights warning](../assets/signature-insights-warning.png)

</p>

## Example

See the [`@metamask/signature-insights-example-snap`](https://github.com/MetaMask/snaps/tree/main/packages/examples/packages/signature-insights)
package for a full example of implementing signature insights.
7 changes: 7 additions & 0 deletions snaps/index.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -123,6 +123,13 @@ the canary distribution of MetaMask:
description: "Add extra friction to the transaction flow if a transaction looks risky.",
flaskOnly: true
},
{
icon: require("./assets/features/signature-insights.png").default,
href: "features/signature-insights",
title: "Signature insights",
description: "Warn users about potentially dangerous signature requests.",
flaskOnly: true
},
{
icon: require("./assets/features/network.png").default,
href: "features/custom-evm-accounts#account-abstraction-erc-4337",
Expand Down
Loading
Loading