Skip to content

Commit

Permalink
feat: sessionStorage for burner wallet (#24)
Browse files Browse the repository at this point in the history
Co-authored-by: Shiv Bhonde <shivbhonde04@gmail.com>
  • Loading branch information
rin-st and technophile-04 authored Dec 20, 2024
1 parent 85c0e37 commit 1849aec
Show file tree
Hide file tree
Showing 10 changed files with 65 additions and 21 deletions.
5 changes: 5 additions & 0 deletions .changeset/dirty-wasps-jam.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"burner-connector": patch
---

feat: sessionStorage for burner wallet
2 changes: 1 addition & 1 deletion CONTRIBUTING.md
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ pnpm run dev

This will start a local server on `http://localhost:3000` with the example app linked to local package

The burner wallet should be automatically connected to sepolia network, and can interact with the [`YourContract`](https://sepolia.etherscan.io/address/0x0D25b202D1B5126ECFcaeFa85f7a37ed86EF79ea) deployed on the sepolia.
The burner wallet should be automatically connected to optimism sepolia network, and can interact with the [`YourContract`](https://optimism-sepolia.blockscout.com/address/0xFB30C0790128b97e3aC540E6124e512E37c47D00).

## Testing with hardhat

Expand Down
12 changes: 12 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,11 @@ pnpm add burner-connector
```ts
import { burner } from "burner-connector";
import { mainnet, base } from "viem/chains";

// burner function can also called with param `{ useSessionStorage: true }` to create a new wallet for each browser tab
// - `useSessionStorage` to false (default) to persist wallet across browser tabs(incognito window will have different wallet)
// - `useSessionStorage` to true to create a new wallet for each browser tab

export const config = createConfig({
chains: [mainnet, base],
connectors: [burner()],
Expand All @@ -49,6 +54,13 @@ import { mainnet, base } from "viem/chains";

const wallets = [metaMaskWallet, rainbowkitBurnerWallet];

// Configure burner wallet storage
// - `useSessionStorage` to false (default) to persist wallet across browser tabs(incognito window will have different wallet)
// - `useSessionStorage` to true to create a new wallet for each browser tab
// rainbowkitBurnerWallet.useSessionStorage = true;

const wallets = [metaMaskWallet, rainbowkitBurnerWallet];

const wagmiConnectors = connectorsForWallets(
[
{
Expand Down
6 changes: 4 additions & 2 deletions example/app/components/Example.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import {
import { FaucetButton } from "./FaucetButton";
import toast from "react-hot-toast";
import deployedContracts from "../contracts/deployedContracts";
import { optimismSepolia } from "viem/chains";

export const Example = () => {
const { isConnected, chain } = useAccount();
Expand All @@ -17,8 +18,9 @@ export const Example = () => {
console.log("isPending", isPending);
const yourContract =
chain?.id && chain.id in deployedContracts
? deployedContracts[chain.id as 11155111 | 31337].YourContract
: deployedContracts["11155111"].YourContract;
? deployedContracts[chain.id as keyof typeof deployedContracts]
.YourContract
: deployedContracts[optimismSepolia.id].YourContract;

const { data: totalCounter } = useReadContract({
...yourContract,
Expand Down
4 changes: 2 additions & 2 deletions example/app/contracts/deployedContracts.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
const deployedContracts = {
11155111: {
11155420: {
YourContract: {
address: "0x0D25b202D1B5126ECFcaeFa85f7a37ed86EF79ea",
address: "0xFB30C0790128b97e3aC540E6124e512E37c47D00",
abi: [
{
inputs: [
Expand Down
9 changes: 6 additions & 3 deletions example/app/wagmiConfig.ts
Original file line number Diff line number Diff line change
@@ -1,10 +1,13 @@
import { connectorsForWallets } from "@rainbow-me/rainbowkit";
import { createConfig } from "wagmi";
import { hardhat, sepolia } from "wagmi/chains";
import { hardhat, optimismSepolia } from "wagmi/chains";
import { metaMaskWallet } from "@rainbow-me/rainbowkit/wallets";
import { createClient, http } from "viem";
import { rainbowkitBurnerWallet } from "burner-connector";

// Use this if you want to enable session storage
// rainbowkitBurnerWallet.useSessionStorage = true;

const wallets = [metaMaskWallet, rainbowkitBurnerWallet];
const walletConnectProjectID = "3a8170812b534d0ff9d794f19a901d64";
const wagmiConnectors = connectorsForWallets(
Expand All @@ -18,10 +21,10 @@ const wagmiConnectors = connectorsForWallets(
{
appName: "scaffold-eth-2",
projectId: walletConnectProjectID,
},
}
);

export const chains = [sepolia, hardhat] as const;
export const chains = [optimismSepolia, hardhat] as const;

export const wagmiConfig = createConfig({
chains: chains,
Expand Down
4 changes: 2 additions & 2 deletions packages/burner-connector/src/burnerConnector/burner.ts
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ export class ChainNotConfiguredError extends BaseError {

type Provider = ReturnType<Transport<"custom", Record<any, any>, EIP1193RequestFn<WalletRpcSchema>>>;

export const burner = () => {
export const burner = ({ useSessionStorage = false }: { useSessionStorage?: boolean } = {}) => {
let connected = true;
let connectedChainId: number;
return createConnector<Provider>((config) => ({
Expand All @@ -55,7 +55,7 @@ export const burner = () => {

const url = chain.rpcUrls.default.http[0];
if (!url) throw new Error("No rpc url found for chain");
const burnerAccount = privateKeyToAccount(loadBurnerPK());
const burnerAccount = privateKeyToAccount(loadBurnerPK({ useSessionStorage }));
const client = createWalletClient({
chain: chain,
account: burnerAccount,
Expand Down
24 changes: 16 additions & 8 deletions packages/burner-connector/src/utils/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,28 +13,36 @@ const isValidPK = (pk: Hex | string | undefined | null): boolean => {
};

/**
* Save the current burner private key to local storage
* Save the current burner private key to storage
*/
export const saveBurnerPK = (privateKey: Hex): void => {
const saveBurnerPK = ({
privateKey,
useSessionStorage = false,
}: {
privateKey: Hex;
useSessionStorage?: boolean;
}): void => {
if (typeof window !== "undefined" && window != null) {
window?.localStorage?.setItem(burnerStorageKey, privateKey);
const storage = useSessionStorage ? window.sessionStorage : window.localStorage;
storage?.setItem(burnerStorageKey, privateKey);
}
};

/**
* Gets the current burner private key from local storage
* Gets the current burner private key from local/session storage
*/
export const loadBurnerPK = (): Hex => {
export const loadBurnerPK = ({ useSessionStorage = false }: { useSessionStorage?: boolean } = {}): Hex => {
let currentSk: Hex = "0x";
if (typeof window !== "undefined" && window != null) {
currentSk = (window?.localStorage?.getItem?.(burnerStorageKey)?.replaceAll('"', "") ?? "0x") as Hex;
const storage = useSessionStorage ? window.sessionStorage : window.localStorage;
currentSk = (storage?.getItem?.(burnerStorageKey)?.replaceAll('"', "") ?? "0x") as Hex;
}

if (!!currentSk && isValidPK(currentSk)) {
return currentSk;
}
// If no burner is found in localstorage, we will generate a random private key
// If no burner is found in storage, we will generate a random private key
const newDefaultPrivateKey = generatePrivateKey();
saveBurnerPK(newDefaultPrivateKey);
saveBurnerPK({ privateKey: newDefaultPrivateKey, useSessionStorage });
return newDefaultPrivateKey;
};
Original file line number Diff line number Diff line change
Expand Up @@ -11,3 +11,10 @@ export const rainbowkitBurnerConnector = (walletDetails: WalletDetailsParams) =>
...walletDetails,
}));
};

export const rainbowkitSessionStorageBurnerConnector = (walletDetails: WalletDetailsParams) => {
return createConnector<Provider>((config) => ({
...burner({ useSessionStorage: true })(config),
...walletDetails,
}));
};
Original file line number Diff line number Diff line change
@@ -1,17 +1,24 @@
import type { Wallet } from "@rainbow-me/rainbowkit";
import { burnerWalletId, burnerWalletName } from "../../utils/index.js";
import { rainbowkitBurnerConnector } from "./rainbowkitBurnerConnector.js";
import { rainbowkitBurnerConnector, rainbowkitSessionStorageBurnerConnector } from "./rainbowkitBurnerConnector.js";

const burnerWalletIconBase64 =
"";

type RainbowkitBurnerWallet = {
(): Wallet;
useSessionStorage?: boolean;
};

/**
* Wagmi config for burner wallet
*/
export const rainbowkitBurnerWallet = (): Wallet => ({
export const rainbowkitBurnerWallet: RainbowkitBurnerWallet = () => ({
id: burnerWalletId,
name: burnerWalletName,
iconUrl: burnerWalletIconBase64,
iconBackground: "#ffffff",
createConnector: rainbowkitBurnerConnector,
createConnector: rainbowkitBurnerWallet.useSessionStorage
? rainbowkitSessionStorageBurnerConnector
: rainbowkitBurnerConnector,
});

0 comments on commit 1849aec

Please sign in to comment.