Loading source
Pulling the file list, source metadata, and syntax-aware rendering for this listing.
Source from repo
Comprehensive Solana development skill covering @solana/kit v5, Anchor programs, LiteSVM testing, and security patterns.
Files
Skill
Size
Entrypoint
Format
Open file
Syntax-highlighted preview of this file as included in the skill package.
references/kit/plugins.md
1---2title: Plugins & Client Composition3description: Solana Kit plugin architecture, all-in-one RPC/LiteSVM plugins, signer plugins, custom client composition, and plugin ordering rules.4---56# Solana Kit Plugins & Client Composition78Kit clients are built by chaining `.use(plugin)` calls onto `createClient()`. Each plugin extends the client with new properties or methods. Plugins that depend on others (e.g., RPC needs a payer) must come after their dependencies — TypeScript enforces this.910## All-in-One Clients1112### Production Client (mainnet/devnet/custom)1314```bash15npm install @solana/kit @solana/kit-plugin-rpc @solana/kit-plugin-signer16```1718```ts19import { createClient } from '@solana/kit';20import { solanaRpc } from '@solana/kit-plugin-rpc';21import { signer } from '@solana/kit-plugin-signer';2223const client = createClient()24.use(signer(mySigner)) // sets payer + identity to the same keypair25.use(solanaRpc({ rpcUrl: 'https://api.mainnet-beta.solana.com' }));2627await client.sendTransaction([myInstruction]);28```2930`solanaRpc` installs an RPC connection, RPC subscriptions, minimum-balance computation, transaction planner, and transaction executor in one call. It requires a `payer` to be set first — `signer()` covers that and the identity role at the same time. Reach for the role-specific `payer()` + `identity()` only when fees and authority must come from different keypairs.3132**`solanaRpc` options:**3334| Option | Type | Description |35|--------|------|-------------|36| `rpcUrl` | `string` | RPC endpoint (required) |37| `rpcSubscriptionsUrl` | `string` | WS endpoint (defaults to `rpcUrl` with `http`→`ws`) |38| `rpcConfig` | `object` | Forwarded to `createSolanaRpc` |39| `rpcSubscriptionsConfig` | `object` | Forwarded to `createSolanaRpcSubscriptions` |40| `transactionConfig` | `object` | Tx planner options (priority fees, etc.) |41| `maxConcurrency` | `number` | Concurrent tx limit (default: 10) |42| `skipPreflight` | `boolean` | Always skip preflight (default: false) |4344### Cluster-Specialized Variants4546```ts47import { createClient } from '@solana/kit';48import { solanaMainnetRpc, solanaDevnetRpc, solanaLocalRpc } from '@solana/kit-plugin-rpc';49import { signer, signerFromFile } from '@solana/kit-plugin-signer';5051// Mainnet — type-narrowed; airdrop is NOT exposed52const main = createClient().use(signer(s)).use(solanaMainnetRpc({ rpcUrl: '...' }));5354// Devnet — defaults to https://api.devnet.solana.com, includes airdrop55const dev = createClient().use(signer(s)).use(solanaDevnetRpc());5657// Local — defaults to http://127.0.0.1:8899, includes airdrop58const local = await createClient()59.use(signerFromFile('~/.config/solana/id.json'))60.use(solanaLocalRpc());61```6263### LiteSVM Test Client6465```bash66npm install @solana/kit @solana/kit-plugin-litesvm @solana/kit-plugin-signer67```6869```ts70import { createClient, lamports } from '@solana/kit';71import { litesvm } from '@solana/kit-plugin-litesvm';72import { airdropSigner, generatedSigner } from '@solana/kit-plugin-signer';7374const client = await createClient()75.use(generatedSigner())76.use(litesvm())77.use(airdropSigner(lamports(1_000_000_000n)));7879client.svm.setAccount(myTestAccount);80client.svm.addProgramFromFile(myProgramAddress, 'program.so');8182await client.sendTransaction([myInstruction]);83```8485`litesvm()` is Node.js only. Browser/React Native builds throw.8687---8889## Client API Surface9091See [overview.md](overview.md#client-api) for the full surface (`client.rpc`, `client.payer`, `client.sendTransaction`, etc.). Plugin-specific additions:9293- `solanaDevnetRpc` / `solanaLocalRpc` / `litesvm` add `client.airdrop(address, lamports)`.94- `litesvm` additionally adds `client.svm` for direct LiteSVM access.95- The low-level composition (Custom Client Composition below) also exposes `client.transactionPlanner` and `client.transactionPlanExecutor` directly.9697---9899## Signer Plugins (`@solana/kit-plugin-signer`)100101Kit clients hold two signer roles:102- **`payer`** — pays fees and rent103- **`identity`** — wallet/authority for application accounts104105In most apps both roles are the same keypair, so **default to the `signer*` variants** — they install one keypair into both slots in one call. Use the role-specific `payer*` / `identity*` variants only when fees and authority come from different keypairs (e.g., gasless flows, treasury accounts, multisig).106107| Variant | Sets |108|---|---|109| `signer*` (recommended default) | both `payer` and `identity` (same keypair) |110| `payer*` | only `payer` |111| `identity*` | only `identity` |112113| Plugin | Behavior |114|---|---|115| `signer(s)` / `payer(s)` / `identity(s)` | Install an existing `TransactionSigner` |116| `generatedSigner()` / `generatedPayer()` / `generatedIdentity()` | Async; generate a new keypair |117| `generatedSignerWithSol(amount)` / `generatedPayerWithSol(amount)` / `generatedIdentityWithSol(amount)` | Async; generate + airdrop. Requires an airdrop function already on the client (for low-level composition, install `rpcAirdrop()` first). |118| `signerFromFile(path)` / `payerFromFile(path)` / `identityFromFile(path)` | Async; load keypair from a JSON file |119| `airdropSigner(amount)` / `airdropPayer(amount)` / `airdropIdentity(amount)` | Airdrop SOL to an already-installed signer. Use with all-in-one `solanaLocalRpc` / `solanaDevnetRpc` / `litesvm` when the RPC plugin needs a payer first. |120121```ts122import { createClient, lamports } from '@solana/kit';123import {124rpcAirdrop,125solanaRpcConnection,126solanaRpcSubscriptionsConnection,127} from '@solana/kit-plugin-rpc';128import { generatedSignerWithSol } from '@solana/kit-plugin-signer';129130// Airdrop function must exist before generatedSignerWithSol131const client = await createClient()132.use(solanaRpcConnection('http://127.0.0.1:8899'))133.use(solanaRpcSubscriptionsConnection('ws://127.0.0.1:8900'))134.use(rpcAirdrop())135.use(generatedSignerWithSol(lamports(10_000_000_000n)));136```137138**Role-split example** (different keypairs for fees vs. authority):139140```ts141import { createClient } from '@solana/kit';142import { solanaDevnetRpc } from '@solana/kit-plugin-rpc';143import { payer, identity } from '@solana/kit-plugin-signer';144145const client = createClient()146.use(payer(feePayerSigner)) // pays fees147.use(identity(walletSigner)) // owns/authorizes accounts148.use(solanaDevnetRpc());149```150151---152153## Custom Client Composition154155When the all-in-one bundles don't fit (custom transaction planner, partial capabilities, third-party services), build the client out of low-level plugins:156157```ts158import { createClient } from '@solana/kit';159import {160rpc,161rpcAirdrop,162rpcGetMinimumBalance,163rpcTransactionPlanner,164rpcTransactionPlanExecutor,165} from '@solana/kit-plugin-rpc';166import { signerFromFile } from '@solana/kit-plugin-signer';167import { planAndSendTransactions } from '@solana/kit-plugin-instruction-plan';168169const client = await createClient()170.use(rpc('https://api.devnet.solana.com')) // adds client.rpc + client.rpcSubscriptions171.use(signerFromFile('path/to/keypair.json')) // adds client.payer + client.identity172.use(rpcAirdrop()) // adds client.airdrop173.use(rpcGetMinimumBalance()) // adds client.getMinimumBalance174.use(rpcTransactionPlanner()) // adds client.transactionPlanner175.use(rpcTransactionPlanExecutor()) // adds client.transactionPlanExecutor176.use(planAndSendTransactions()); // adds client.sendTransaction(s)177```178179### Plugin Ordering180181Plugins that depend on others must come after their dependencies. TypeScript enforces this:182183```ts184// ✅ Correct — signer + rpc before planner/executor185createClient()186.use(signer(mySigner))187.use(rpc(url))188.use(rpcTransactionPlanner())189.use(planAndSendTransactions());190191// ❌ Type error — solanaRpc requires payer192createClient()193.use(solanaRpc({ rpcUrl: url }))194.use(signer(mySigner));195```196197### Async Plugins198199Some plugins are async (e.g., `signerFromFile`, `generatedSigner`, `generatedSignerWithSol`). The `.use()` method handles awaiting automatically — `await` the final client:200201```ts202const client = await createClient()203.use(signerFromFile('./keypair.json')) // async204.use(solanaLocalRpc());205```206207---208209## Plugin Catalog210211### Official Plugins212213| Package | Plugins | Purpose |214|---------|---------|---------|215| `@solana/kit-plugin-rpc` | `solanaRpc`, `solanaMainnetRpc`, `solanaDevnetRpc`, `solanaLocalRpc`, `rpc`, `solanaRpcConnection`, `rpcAirdrop`, `rpcGetMinimumBalance`, `rpcTransactionPlanner`, `rpcTransactionPlanExecutor` | RPC connectivity + tx planning/execution |216| `@solana/kit-plugin-signer` | `signer*` (default — sets both roles), `payer*`, `identity*` (role-specific); each comes in plain, `generated*`, `*WithSol`, `*FromFile`, and `airdrop*` forms | Signer management |217| `@solana/kit-plugin-litesvm` | `litesvm`, `litesvmConnection`, `litesvmAirdrop`, `litesvmTransactionPlanner`, `litesvmTransactionPlanExecutor` | In-memory test environment |218| `@solana/kit-plugin-airdrop` | `airdrop`, `rpcAirdrop` | SOL faucet (typically pulled in transitively) |219| `@solana/kit-plugin-instruction-plan` | `planAndSendTransactions` | Instruction batching + sending sugar |220221### Program Plugins222223Codama-generated `@solana-program/*` packages also export program plugins that attach fluent APIs to the client:224225```ts226import { createClient } from '@solana/kit';227import { solanaLocalRpc } from '@solana/kit-plugin-rpc';228import { signerFromFile } from '@solana/kit-plugin-signer';229import { tokenProgram } from '@solana-program/token';230231const client = await createClient()232.use(signerFromFile('~/.config/solana/id.json'))233.use(solanaLocalRpc())234.use(tokenProgram());235236// Fluent API — auto-derives ATAs, defaults payer from client237await client.token.instructions238.transferToATA({ mint, authority: ownerSigner, recipient, amount: 50n, decimals: 2 })239.sendTransaction();240```241242| Package | Plugin | Adds |243|---------|--------|------|244| `@solana-program/token` | `tokenProgram()` | `client.token.instructions` (createMint, mintToATA, transferToATA, etc.) |245246### Example Implementations247248| Package | Exports | Purpose | Code Example |249|---------|---------|---------|--------------|250| `@solana/kora` | `createKitKoraClient`, `koraPlugin` | Gasless transactions | https://github.com/solana-foundation/kora/blob/main/sdks/ts/src/kit/index.ts |251252---253254## Building Custom Plugins255256See [advanced.md](advanced.md) for the full guide on authoring plugins and assembling domain-specific clients.257258A plugin is a function that takes a client and returns a new one:259260```ts261type ClientPlugin<TInput extends object, TOutput extends Promise<object> | object> =262(input: TInput) => TOutput;263```264265Quick example:266267```ts268function myCustomPlugin() {269return <T extends object>(client: T) => ({270...client,271myMethod: () => console.log('hello'),272});273}274275const client = createClient().use(myCustomPlugin());276client.myMethod(); // 'hello'277```278279Plugins can require capabilities from previous plugins:280281```ts282function myRpcPlugin() {283return <T extends { rpc: SolanaRpc }>(client: T) => ({284...client,285fetchBalance: (addr: Address) => client.rpc.getBalance(addr).send(),286});287}288289// ✅ Works — rpc installed first290createClient().use(rpc(url)).use(myRpcPlugin());291292// ❌ Type error — rpc not present293createClient().use(myRpcPlugin());294```295