Kit

Generate a signer

Create a new keypair signer and airdrop lamports to it

Before we can start crafting transactions, we're going to need a wallet that can sign transactions and pay for fees. In this article, we'll explore how to generate new keypairs and how to airdrop some lamports to them in test environments.

Generate a new CryptoKeypair

The Kit library uses the native Crypto API for cryptographic tasks like generating key pairs and signing or verifying messages. This improves security by leveraging the browser's built-in functions instead of third-party libraries.

To create a new keypair, you can use the generateKeyPair helper function. This will return an instance of the native CryptoKeyPair type.

import {  } from "@solana/kit";
 
const : CryptoKeyPair = await ();

And just like that, we have a wallet that can be used to sign transactions. However, Kit provides a useful abstraction on top of anything that can sign messages and/or transactions called Signers.

Generate a new KeyPairSigner

Using CryptoKeyPairs is one way to sign and verify transactions but it isn't the only way. What if I want my transactions to be signed by an in-browser wallet? What if I want to sign them using a hardware wallet instead? What about a multisig wallet? Or what if I want some custom logic that detects the best way to sign a transaction based on the current environment?

The Signer API allows us to decouple the signing logic from the rest of our code by providing signer interfaces that can be leveraged throughout the library. A Signer is an object that holds an Address and implements its own signing logic for messages and transactions.

Different types of signers are available for various use cases, but in this tutorial, we'll focus on the KeyPairSigner which signs transactions and messages using a CryptoKeyPair. It implements the TransactionSigner and MessageSigner interfaces so any function that requires a transaction or message signer will be able to use it.

We can use the generateKeyPairSigner function to create a new KeyPairSigner instance like so.

import {  } from "@solana/kit";
 
const  = await ();

As you can see, this is very similar to generating a CryptoKeyPair, but now we have an abstracted Signer object that can be replaced with any other type of signer in the future without affecting the rest of our code.

For instance, here's how we could replace our KeyPairSigner with a signer that uses an in-browser wallet instead.

import { , ,  } from "@solana/kit";
import {  } from "@solana/react";
 
let : ;
 = await (); 
 = (, ); 

For a deeper dive into the Signer API, check out this in-depth article about signers.

Airdrop lamports

Now that we know how to generate a new signer, let's make sure we give it some lamports so it can pay for transaction fees and account storage. For that we can use the airdropFactory function. Given an RPC and RPC subscriptions object, it will return a function that can be used to airdrop lamports to any address — provided we are in a test environment such as localnet.

import {
  ,
  ,
  ,
  ,
  ,
} from "@solana/kit";
 
// Create the RPC, RPC Subscriptions and airdrop function.
const  = ("http://127.0.0.1:8899");
const  = ("ws://127.0.0.1:8900");
const  = ({ ,  });
 
// Generate a new wallet with 1 SOL.
const  = await ();
await ({
  : .,
  : (1_000_000_000n),
  : "confirmed",
});

Adjust our Client object

Since we're going to rely on this signer for the rest of the tutorial, let's adjust our Client object to include it.

src/client.ts
import { , , , ,  } from "@solana/kit";
 
export type  = {
  : <>;
  : <>;
  :  & ;
};
 
let :  | undefined;
export async function (): <> {
  if (!) {
    // Create RPC objects and airdrop function.
    const  = ("http://127.0.0.1:8899");
    const  = ("ws://127.0.0.1:8900");
    const  = ({ ,  });
 
    // Create a wallet with lamports.
    const  = await ();
    await ({
      : .,
      : (1_000_000_000n),
      : "confirmed",
    });
 
    // Store the client.
     = { , ,  };
  }
  return ;
}

Notice how we had to make the createClient function asynchronous since it now contains asynchronous code to generate the KeyPairSigner and airdrop lamports to it. That means, we also need to adjust our main tutorial function accordingly. And whilst we're at it, let's display the balance of our newly created wallet.

src/index.ts
async function () {
  const  = await ();
  const { :  } = await ..(..).();
  .(`Balance: ${} lamports.`);
}

We've now got everything we need to start crafting transactions! In the next article, we'll build instructions that create a new token on Solana.

On this page