BOB Bitcoin MetaMask Snap
MetaMask Snaps allow us to add features and functionality to the standard MetaMask wallet. The BOB Bitcoin Snap is a MetaMask Snap that allows you to use MetaMask to interact with Bitcoin on the BOB network.
Features
Use MetaMask to:
- Send Bitcoin
- Receive Bitcoin
- Cache the extended public key
- Sign Bitcoin transactions
- Inscribe Ordinals (text and images) including BRC20s
- Send inscriptions (including BRC20s)
tip
Take a look at the BTC Snap source code and our MetaMask Snap demo application
Usage
Connecting to the BOB BTC Snap
tip
To see how to connect to the BOB BTC Snap and call the available methods, take a look at the utils.ts file in our demo app.
import { MetaMaskInpageProvider } from "@metamask/providers";
declare global {
interface Window {
ethereum: MetaMaskInpageProvider;
}
}
const { ethereum } = window;
const snapId = "npm:@gobob/btcsnap";
export async function checkConnection(): Promise<boolean> {
const snaps = await ethereum.request({
method: "wallet_getSnaps",
});
const hasMySnap = Object.keys(snaps || []).includes(snapId);
return hasMySnap;
}
export async function connect(cb: (connected: boolean) => void) {
let connected = false;
try {
const result: any = await ethereum.request({
method: "wallet_requestSnaps",
params: {
[snapId]: {},
},
});
const hasError = !!result?.snaps?.[snapId]?.error;
connected = !hasError;
} finally {
cb(connected);
}
}
Getting the extended public key
export enum BitcoinNetwork {
Main = "mainnet",
Test = "testnet",
}
export enum BitcoinScriptType {
P2PKH = "P2PKH",
P2SH_P2WPKH = "P2SH-P2WPKH",
P2WPKH = "P2WPKH",
}
export interface ExtendedPublicKey {
xpub: string;
mfp: string;
}
export async function getExtendedPublicKey(
network: BitcoinNetwork,
scriptType: BitcoinScriptType
): Promise<ExtendedPublicKey> {
const networkParams = network === BitcoinNetwork.Main ? "main" : "test";
try {
return (await ethereum.request({
method: "wallet_invokeSnap",
params: {
snapId,
request: {
method: "btc_getPublicExtendedKey",
params: {
network: networkParams,
scriptType,
},
},
},
})) as ExtendedPublicKey;
} catch (err: any) {
const error = new SnapError(
err?.message || "Get extended public key failed"
);
console.error(error);
throw error;
}
}
Using the BOB BTC Snap in a React application
tip
Take a look at the UI code in our demo application to see how this hook can be used.
import { useCallback, useEffect, useState } from "react";
import { addressFromExtPubKey } from "../utils/btcsnap-signer";
import {
BitcoinNetwork,
BitcoinScriptType,
checkConnection,
connect,
getExtendedPublicKey,
} from "../utils/btcsnap-utils";
import { useLocalStorage, LocalStorageKey } from "./useLocalStorage";
import { useGetInscriptionIds } from "./useGetInscriptionIds";
import { useQueryClient } from "@tanstack/react-query";
import { BITCOIN_NETWORK } from "../utils/config";
const bitcoinNetwork =
BITCOIN_NETWORK === "mainnet" ? BitcoinNetwork.Main : BitcoinNetwork.Test;
const getDerivedBtcAddress = async () => {
const xpub = await getExtendedPublicKey(
bitcoinNetwork,
BitcoinScriptType.P2WPKH
);
const bitcoinAddress = addressFromExtPubKey(xpub.xpub, bitcoinNetwork)!;
return {
bitcoinAddress,
};
};
const connectionCheck = async () => {
const isConnected = await checkConnection();
return isConnected;
};
const useBtcSnap = () => {
const [isConnected, setIsConnected] = useState<boolean>(false);
const queryClient = useQueryClient();
const [bitcoinAddress, setBitcoinAddress, removeBitcoinAddress] =
useLocalStorage(LocalStorageKey.DERIVED_BTC_ADDRESS);
const { refetch } = useGetInscriptionIds(bitcoinAddress);
useEffect(() => {
if (!bitcoinAddress) return;
refetch();
}, [bitcoinAddress, refetch]);
const connectBtcSnap = useCallback(async () => {
connect(async (connected: boolean) => {
if (connected) {
const { bitcoinAddress } = await getDerivedBtcAddress();
setBitcoinAddress(bitcoinAddress);
}
});
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [setBitcoinAddress]);
useEffect(() => {
const checkConnection = async () => {
const connected = await connectionCheck();
// This will reset BTC address if user has disconnected
if (!connected && bitcoinAddress) {
removeBitcoinAddress();
queryClient.removeQueries();
}
setIsConnected(connected);
};
checkConnection();
}, [
bitcoinAddress,
isConnected,
queryClient,
removeBitcoinAddress,
setBitcoinAddress,
]);
return { connectBtcSnap, bitcoinAddress, isConnected };
};
export { useBtcSnap };