diff --git a/.changeset/dirty-wasps-jam.md b/.changeset/dirty-wasps-jam.md new file mode 100644 index 0000000..50686ec --- /dev/null +++ b/.changeset/dirty-wasps-jam.md @@ -0,0 +1,5 @@ +--- +"burner-connector": patch +--- + +feat: sessionStorage for burner wallet diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 90b779f..7f7debe 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -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 diff --git a/README.md b/README.md index 18601ce..7cdc9c0 100644 --- a/README.md +++ b/README.md @@ -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()], @@ -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( [ { diff --git a/example/app/components/Example.tsx b/example/app/components/Example.tsx index 56da377..f7de96c 100644 --- a/example/app/components/Example.tsx +++ b/example/app/components/Example.tsx @@ -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(); @@ -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, diff --git a/example/app/contracts/deployedContracts.ts b/example/app/contracts/deployedContracts.ts index 1431689..bc5b694 100644 --- a/example/app/contracts/deployedContracts.ts +++ b/example/app/contracts/deployedContracts.ts @@ -1,7 +1,7 @@ const deployedContracts = { - 11155111: { + 11155420: { YourContract: { - address: "0x0D25b202D1B5126ECFcaeFa85f7a37ed86EF79ea", + address: "0xFB30C0790128b97e3aC540E6124e512E37c47D00", abi: [ { inputs: [ diff --git a/example/app/wagmiConfig.ts b/example/app/wagmiConfig.ts index c9e20c1..9a0f45d 100644 --- a/example/app/wagmiConfig.ts +++ b/example/app/wagmiConfig.ts @@ -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( @@ -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, diff --git a/packages/burner-connector/src/burnerConnector/burner.ts b/packages/burner-connector/src/burnerConnector/burner.ts index fc17f23..0d0e472 100644 --- a/packages/burner-connector/src/burnerConnector/burner.ts +++ b/packages/burner-connector/src/burnerConnector/burner.ts @@ -30,7 +30,7 @@ export class ChainNotConfiguredError extends BaseError { type Provider = ReturnType, EIP1193RequestFn>>; -export const burner = () => { +export const burner = ({ useSessionStorage = false }: { useSessionStorage?: boolean } = {}) => { let connected = true; let connectedChainId: number; return createConnector((config) => ({ @@ -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, diff --git a/packages/burner-connector/src/utils/index.ts b/packages/burner-connector/src/utils/index.ts index 053278b..6b0e754 100644 --- a/packages/burner-connector/src/utils/index.ts +++ b/packages/burner-connector/src/utils/index.ts @@ -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; }; diff --git a/packages/burner-connector/src/wallets/rainbowkit/rainbowkitBurnerConnector.ts b/packages/burner-connector/src/wallets/rainbowkit/rainbowkitBurnerConnector.ts index b7dc7fa..cd54000 100644 --- a/packages/burner-connector/src/wallets/rainbowkit/rainbowkitBurnerConnector.ts +++ b/packages/burner-connector/src/wallets/rainbowkit/rainbowkitBurnerConnector.ts @@ -11,3 +11,10 @@ export const rainbowkitBurnerConnector = (walletDetails: WalletDetailsParams) => ...walletDetails, })); }; + +export const rainbowkitSessionStorageBurnerConnector = (walletDetails: WalletDetailsParams) => { + return createConnector((config) => ({ + ...burner({ useSessionStorage: true })(config), + ...walletDetails, + })); +}; diff --git a/packages/burner-connector/src/wallets/rainbowkit/rainbowkitBurnerWallet.ts b/packages/burner-connector/src/wallets/rainbowkit/rainbowkitBurnerWallet.ts index 7163624..4b51767 100644 --- a/packages/burner-connector/src/wallets/rainbowkit/rainbowkitBurnerWallet.ts +++ b/packages/burner-connector/src/wallets/rainbowkit/rainbowkitBurnerWallet.ts @@ -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, });