Skip to content

Commit

Permalink
feat: add metamask sdk
Browse files Browse the repository at this point in the history
  • Loading branch information
EdouardBougon committed Dec 12, 2024
1 parent f73b2cd commit 0108e92
Show file tree
Hide file tree
Showing 7 changed files with 843 additions and 20 deletions.
3 changes: 2 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,7 @@
"@gnosis.pm/safe-apps-web3-react": "^1.2.0",
"@heroicons/react": "^1.0.6",
"@lingui/react": "^3.14.0",
"@metamask/sdk": "^0.31.2",
"@mui/icons-material": "^5.10.14",
"@mui/lab": "5.0.0-alpha.103",
"@mui/material": "^5.10.9",
Expand Down Expand Up @@ -158,4 +159,4 @@
"budgetPercentIncreaseRed": 20,
"showDetails": true
}
}
}
32 changes: 32 additions & 0 deletions public/icons/wallets/metaMask.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
14 changes: 14 additions & 0 deletions src/components/WalletConnection/WalletSelector.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,15 @@ const WalletRow = ({ walletName, walletType }: WalletRowProps) => {
alt={`browser wallet icon`}
/>
);
case WalletType.METAMASK_SDK:
return (
<img
src={`/icons/wallets/metaMask.svg`}
width="24px"
height="24px"
alt={`browser wallet icon`}
/>
);
case WalletType.WALLET_LINK:
return (
<img
Expand Down Expand Up @@ -177,6 +186,11 @@ export const WalletSelector = () => {
walletName="Browser wallet"
walletType={WalletType.INJECTED}
/>
<WalletRow
key="metamasksdk_wallet"
walletName="MetaMask"
walletType={WalletType.METAMASK_SDK}
/>
<WalletRow
key="walletconnect_wallet"
walletName="WalletConnect"
Expand Down
274 changes: 274 additions & 0 deletions src/libs/web3-data-provider/MetaMaskSDKConnector.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,274 @@
/* eslint-disable @typescript-eslint/no-explicit-any */
import type {
MetaMaskSDK as _MetaMaskSDK,
MetaMaskSDKOptions as _MetaMaskSDKOptions,
SDKProvider,
} from '@metamask/sdk';
import { AbstractConnector } from '@web3-react/abstract-connector';
import { ConnectorUpdate } from '@web3-react/types';

type Address = string;

/**
* MetaMaskSDK options.
*/
export type MetaMaskSDKConnectorOptions = Pick<
_MetaMaskSDKOptions,
'infuraAPIKey' | 'readonlyRPCMap' | 'headless'
> & {
dappMetadata: Pick<_MetaMaskSDKOptions['dappMetadata'], 'name' | 'url' | 'iconUrl'>;
supportedChainIds?: number[];
};

/**
* Listener type for MetaMaskSDK events.
*/
type Listener = Parameters<AbstractConnector['on']>[1];

/**
* Error thrown when the MetaMaskSDK is not installed.
*/
export class NoMetaMaskSDKError extends Error {
public constructor() {
super('MetaMaskSDK not installed');
this.name = NoMetaMaskSDKError.name;
Object.setPrototypeOf(this, NoMetaMaskSDKError.prototype);
}
}

/**
* Parses a chainId from a string or number.
*/
function parseChainId(chainId: string | number) {
return typeof chainId === 'number'
? chainId
: Number.parseInt(chainId, chainId.startsWith('0x') ? 16 : 10);
}

/**
* MetaMask SDK Connector
*/
export class MetaMaskSDKConnector extends AbstractConnector {
private readonly options: Omit<MetaMaskSDKConnectorOptions, 'supportedChainIds'>;

public sdk: _MetaMaskSDK | undefined;
private provider: SDKProvider | undefined;

constructor({ supportedChainIds, ...options }: MetaMaskSDKConnectorOptions) {
super({ supportedChainIds: supportedChainIds });

const defaultUrl =
typeof window !== 'undefined' ? `${window.location.protocol}//${window.location.host}` : '';

this.options = {
...options,
dappMetadata: options?.dappMetadata ?? {
url: defaultUrl,
name: defaultUrl !== '' ? undefined : 'web3-react',
},
};

this.onConnect = this.onConnect.bind(this);
this.onDisconnect = this.onDisconnect.bind(this);
this.onChainChanged = this.onChainChanged.bind(this);
this.onAccountsChanged = this.onAccountsChanged.bind(this);
}

/**
* Indicates whether the user is connected to the MetaMaskSDK.
*/
public async isConnected() {
try {
if (this.provider?.isConnected?.() === true) {
if (this.sdk?.isExtensionActive() === true) {
const accounts = await this.getAccounts();
return accounts.length > 0;
}

return true;
}
} catch {
// ignore
}

return false;
}

/**
* Activate
*/
public async activate(): Promise<ConnectorUpdate> {
if (this.sdk && this.provider) {
const isConnected = await this.isConnected();

if (!isConnected) {
await this.sdk.connect();
}

return { provider: this.provider, account: await this.getAccount() };
}

const MetaMaskSDK = await import('@metamask/sdk').then((m) => m?.default ?? m);
this.sdk = new MetaMaskSDK({
_source: 'web3-react',
useDeeplink: true,
injectProvider: false,
forceInjectProvider: false,
forceDeleteProvider: false,
...this.options,
});

await this.sdk.init();

this.provider = this.sdk.getProvider()!;

if (!this.provider) return {};

this.provider.on('connect', this.onConnect as Listener);
this.provider.on('disconnect', this.onDisconnect as Listener);
this.provider.on('chainChanged', this.onChainChanged as Listener);
this.provider.on('accountsChanged', this.onAccountsChanged as Listener);

return await this.activate();
}

/**
* Get MetaMask Provider
*/
public async getProvider(): Promise<SDKProvider> {
if (!this.provider) throw new NoMetaMaskSDKError();

return this.provider;
}

/**
* Get chain id
*/
public async getChainId(): Promise<number> {
const provider = await this.getProvider();
const chainId =
provider.getChainId() ?? (await provider?.request<string>({ method: 'eth_chainId' }));

return parseChainId(chainId);
}

/**
* Get selected account
*/
public async getAccount(): Promise<null | Address> {
const accounts = await this.getAccounts();

return accounts?.[0] ?? null;
}

/**
* Get selected accounts
*/
public async getAccounts(): Promise<Address[]> {
if (!this.provider) {
return [];
}

const accounts = (await this.provider.request<Address[]>({
method: 'eth_accounts',
})) as Address[];

return accounts ?? [];
}

/**
* Deactivate this provider instance, without closing the connection
*/
public deactivate() {
if (!this.provider) {
return;
}

this.provider.removeListener('connect', this.onConnect);
this.provider.removeListener('disconnect', this.onDisconnect);
this.provider.removeListener('chainChanged', this.onChainChanged);
this.provider.removeListener('accountsChanged', this.onAccountsChanged);
}

/**
* Close
*/
public async close() {
this.sdk?.terminate();
this.emitDeactivate();
}

/**
* On connect event handler
*/
private onConnect({ chainId }: { chainId: number | string }): void {
this.emitUpdate({ chainId: parseChainId(chainId) });
}

/**
* On disconnect event handler
*/
private async onDisconnect(error: any): Promise<void> {
const originalError = (error.data as any)?.originalError ?? error;

// If MetaMask emits a `code: 1013` error, wait for reconnection before disconnecting
// https://github.com/MetaMask/providers/pull/120
if (error && originalError.code === 1013 && this.provider) {
const accounts = await this.provider.request<Address[]>({ method: 'eth_accounts' });
if (accounts && accounts.length > 0) return;
}

this.clearCache();

if (error) {
this.emitError(error);
}

// this.emitUpdate({ provider: null, account: null })
}

/**
* On chainChanged event handler
*/
private onChainChanged(chainId: string): void {
this.emitUpdate({ chainId: parseChainId(chainId) });
}

/**
* On accountsChanged event handler
*/
private onAccountsChanged(accounts: string[]): void {
// Disconnect if there are no accounts
if (accounts.length === 0) {
// ... and using browser extension
if (this.sdk?.isExtensionActive()) this.close();
// FIXME(upstream): Mobile app sometimes emits invalid `accountsChanged` event with empty accounts array
else return;
} else {
this.emitUpdate({ account: accounts[0] });
}
}

/**
* Is application authorized
*/
public async isAuthorized() {
try {
const accounts = await this.getAccounts();

return accounts && accounts.length > 0;
} catch {
return false;
}
}

/**
* Clears the cache.
*/
private clearCache() {
localStorage.removeItem('.MMSDK_cached_address');
localStorage.removeItem('.MMSDK_cached_chainId');
localStorage.removeItem('.sdk-comm');
localStorage.removeItem('.MetaMaskSDKLng');
}
}
16 changes: 15 additions & 1 deletion src/libs/web3-data-provider/WalletOptions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,12 +9,14 @@ import { ConnectorUpdate } from '@web3-react/types';
import { WalletLinkConnector } from '@web3-react/walletlink-connector';
import { getNetworkConfig } from 'src/utils/marketsAndNetworksConfig';

import { MetaMaskSDKConnector } from './MetaMaskSDKConnector';
// import { LedgerHQFrameConnector } from 'web3-ledgerhq-frame-connector';
import { WalletConnectConnector } from './WalletConnectConnector';

export enum WalletType {
INJECTED = 'injected',
WALLET_CONNECT = 'wallet_connect',
METAMASK_SDK = 'metamask_sdk',
WALLET_LINK = 'wallet_link',
TORUS = 'torus',
FRAME = 'frame',
Expand Down Expand Up @@ -77,15 +79,27 @@ export const getWallet = (
chainId: ChainId = ChainId.mainnet,
currentChainId: ChainId = ChainId.mainnet
): AbstractConnector => {
const networkConfig = getNetworkConfig(chainId);

switch (wallet) {
case WalletType.READ_ONLY_MODE:
return new ReadOnlyModeConnector();
// case WalletType.LEDGER:
// return new LedgerHQFrameConnector({});
case WalletType.INJECTED:
return new InjectedConnector({});
case WalletType.METAMASK_SDK:
return new MetaMaskSDKConnector({
dappMetadata: {
name: APP_NAME,
iconUrl: APP_LOGO_URL,
},
readonlyRPCMap: {
[`0x${chainId.toString(16)}`]:
networkConfig.privateJsonRPCUrl || networkConfig.publicJsonRPCUrl[0],
},
});
case WalletType.WALLET_LINK:
const networkConfig = getNetworkConfig(chainId);
return new WalletLinkConnector({
appName: APP_NAME,
appLogoUrl: APP_LOGO_URL,
Expand Down
Loading

0 comments on commit 0108e92

Please sign in to comment.