Skip to content

Importing an embedded wallet

Privy enables your users to import a private key into their embedded wallet.

This allows users to bring an existing wallet into your application, enabling them to manage their assets directly within your app. For example, a user can import their MetaMask wallet private key to access and manage their assets within your app seamlessly. This method can also be used by developers to migrate users from a different embedded wallet provider to Privy.

INFO

Privy's React SDK currently supports importing wallets on EVM networks and Solana.

Importing a wallet by private key

To have your user import a private key into their embedded wallet, use the importWallet method from the useImportWallet hook. As a parameter to this method, pass an object containing the user's privateKey as a string.

tsx
import {usePrivy, useImportWallet} from '@privy-io/react-auth';

function ImportWalletButton() {
  const {ready, authenticated} = usePrivy();
  const {importWallet} = useImportWallet();
  const [privateKey, setPrivateKey] = useState('');

  const handleImport = async () => {
    try {
      const wallet = await importWallet({privateKey: privateKey});
      console.log('Wallet imported successfully:', wallet);
    } catch (error) {
      console.error('Failed to import wallet:', error);
    }
  };

  // Check that your user is authenticated
  const isAuthenticated = ready && authenticated;

  return (
    <div>
      <input
        type="text"
        value={privateKey}
        onChange={(e) => setPrivateKey(e.target.value)}
        placeholder="Enter your private key"
      />
      <button onClick={handleImport} disabled={!isAuthenticated}>
        Import my wallet
      </button>
    </div>
  );
}

If importWallet succeeds, the imported wallet will be added to the user's linkedAccounts and the wallets array returned by the useWallets hook, with an imported field that is marked true. As an example, you can find the imported wallet like so:

tsx
import {useWallets} from '@privy-io/react-auth';

function ImportedWalletInfo() {
  const {wallets} = useWallets();
  const importedWallet = wallets.find((wallet) => wallet.imported);
  return (
    <div>
      <h3>Imported Wallet</h3>
      {importedWallet ? <p>Address: {importedWallet.address}</p> : <p>No imported wallet found</p>}
    </div>
  );
}

This method will error if the provided private key is not valid or if the user has previously imported a wallet by private key.

Using imported EVM embedded wallets

You can request signatures and transactions from imported wallets using Privy's native methods, the wallet's EIP1193 provider, or a third-party library like viem or ethers.

Using Privy's native signature and transaction methods

To use Privy's native signMessage, signTypedData, and sendTransaction methods with an imported wallet, simply pass the address for the imported wallet as the final optional parameter to these methods:

tsx
const {signMessage} = usePrivy();
const signature = await signMessage(
  'insert-message-to-sign',
  insertOptionalUIConfigOrUndefined,
  'insert-imported-wallet-address', // Replace with the imported wallet address in EIP55 format
);

Using the EIP1193 provider, viem, and ethers

You can also request signatures and transactions from an imported wallet using the wallet's EIP1193 provider or a library like viem or ethers.

To get the EIP1193 provider for the imported wallet, first find the corresponding ConnectedWallet object from the useWallets array:

tsx
const {wallets} = useWallets();
// `ConnectedWallet.imported` is set to `true` for imported wallets
const importedWallet = wallets.find(
  (wallet) => wallet.walletClientType === 'privy' && wallet.imported,
);

Then, call the object's getEthereumProvider method to get an EIP1193 provider for that wallet:

tsx
const provider = await importedWallet.getEthereumProvider();

You can then easily pass that EIP1193 provider to a library like viem or ethers to use those libraries' interfaces to send requests to the wallet.

Using imported Solana embedded wallets

You can request signatures and send transactions using your imported Solana wallet just like with regular embedded Solana wallets.

tsx
const {wallets} = useSolanaWallets();
// `ConnectedWallet.imported` is set to `true` for imported wallets
const importedWallet = wallets.find(
  (wallet) => wallet.walletClientType === 'privy' && wallet.imported,
);

Then, call the object's signMessage method to sign.

tsx
const signature = await importedWallet.sign('Sign this message!');

Exporting imported embedded wallets

Privy enables your users to export the full private key for their embedded wallet, even if it was imported. This allows them to use their embedded wallet address with another wallet client, such as MetaMask in any application.

To have your user export their embedded wallet's private key, use the exportWallet method from the usePrivy hook. As a parameter to this method, you must pass an object containing an address field with the user's imported wallet address as a string.

As an example, you might have an export wallet button in your app like so:

tsx
import {usePrivy, useWallets} from '@privy-io/react-auth';

function ExportWalletButton() {
  const {ready, authenticated, exportWallet} = usePrivy();
  const {wallets} = useWallets();
  // Check that your user is authenticated
  const isAuthenticated = ready && authenticated;
  // Find your user's imported wallet
  const importedWallet = wallets.find(
    (wallet) => wallet.walletClientType === 'privy' && wallet.imported,
  );

  const exportImportedWallet = async () => {
    if (!importedWallet) return;
    await exportWallet({address: importedWallet.address});
  };

  return (
    <button onClick={exportImportedWallet} disabled={!isAuthenticated || !importedWallet}>
      Export my wallet
    </button>
  );
}

INFO

Exporting the seed phrase for an imported embedded wallet is not supported, as it is impossible to recover the full seed phrase for the wallet from the imported private key.