Skip to main content
  1. Use get_account_interface instead of get_account to store AccountInterface.
  2. Use the AMM’s LightProgramInterface trait to load inactive markets when needed.

Step 1: Use get_account_interface

get_account_interface is a new RPC endpoint that returns a superset of get_account. AccountInterface stores additional info in Option<ColdContext> that you will need later.

Account

pub struct Account {
    pub lamports: u64,
    pub data: Vec<u8>,
    pub owner: Pubkey,
    pub executable: bool,
    pub rent_epoch: Epoch,
}

AccountInterface

pub struct AccountInterface {
   key: Pubkey,
   account: Account,
   cold: Option<ColdContext>
}

Step 2: Use the LightProgramInterface trait

All rent-free programs expose a LightProgramInterface trait in their SDK. Using this trait allows you to:
  1. Maintain a cache of &[AccountInterface]
  2. Load inactive “cold” accounts back into the onchain account space when building Swap transactions in a standardized way.
// AMM SDK implements this.
pub trait LightProgramInterface {
    fn from_keyed_accounts(accounts: &[AccountInterface]) -> Result<Self, Self::Error>;
    fn get_accounts_to_update(&self, ix: &Self::Instruction) -> Vec<AccountToFetch>;
    fn update(&mut self, accounts: &[AccountInterface]) -> Result<(), Self::Error>;
    fn get_specs_for_instruction(&self, ix: &Self::Instruction) -> Vec<AccountSpec<Self::Variant>>;
}

Step 3: Load cold accounts when building Swap instructions

When building Swap instructions, prepend a create_load_instructions call. This only adds latency if markets are cold.
// add dynamic load instructions if cold
let specs = sdk.get_specs_for_instruction(&ExampleAmmSdk::LightInstruction::Swap);
if specs.iter().any(|s| s.is_cold()) {
    let load_ixs = create_load_instructions(
        &specs,
        payer.pubkey(),
        sdk.light_config_pda(),
        sdk.light_rent_sponsor_pda(),
        &rpc,
    ).await?;
    instructions.extend(load_ixs);
}

// add swap instruction...

Full Example

Dependencies

[dependencies]
light-client = {version = "0.18.0", features = ["v2"]}

# Example Program SDK that implements LightProgramInterface (provided by AMM team)
example-amm-sdk = "0.1"

Code

use light_client::interface::{
    create_load_instructions, LightProgramInterface, AccountSpec,
};
use example_amm_sdk::{ExampleAmmSdk};

// 1. Fetch account interfaces (works for both hot and cold)
let pool_interface = rpc
    .get_account_interface(&pool_address, &ExampleAmmSdk::program_id())
    .await?;

// 2. Initialize SDK from interfaces
let mut sdk = ExampleAmmSdk::from_keyed_accounts(&[pool_interface])?;

// 3. Fetch related accounts and update SDK state
let accounts_to_fetch = sdk.get_accounts_to_update(&ExampleAmmSdk::LightInstruction::Swap);
let keyed_accounts = rpc.get_multiple_account_interfaces(&accounts_to_fetch).await?;
sdk.update(&keyed_accounts)?;

// 4. Quote (works same for hot or cold)
let quote = sdk.quote(amount_in, min_out)?;

// 5. Build transaction
let mut ixs = vec![];

// Prepend load instructions if any accounts are cold
let specs = sdk.get_specs_for_instruction(&ExampleAmmSdk::LightInstruction::Swap);
if specs.iter().any(|s| s.is_cold()) {
    let load_ixs = create_load_instructions(
        &specs,
        payer.pubkey(),
        sdk.light_config_pda(),           
        sdk.light_rent_sponsor_pda(),
        &rpc,
    ).await?;
    ixs.extend(load_ixs);
}

// Add actual swap instruction
ixs.push(sdk.swap_ix(&swap_params)?);

// 6. Send
rpc.send_transaction(&ixs, &payer).await?;

Key Types

TypeSourcePurpose
Rpc traitlight-clientRPC client with get_account_interface methods
AccountInterfacelight-clientUnified hot/cold account type
LightProgramInterfacelight-clientTrait that program SDKs implement
AccountSpeclight-clientSpecifies account load requirements

Reference Implementation

ResourceLink
AMM Programcp-swap-reference
LightProgramInterface Trait ImplCpSwapSdk
Client Testprogram.rs

Hot vs Cold

HotCold
On-chainYesLedger (compressed)
QuoteWorksWorks
SwapDirectLoad first / Bundle
LatencyNormal+0-200ms*
Tx sizeNormal+100-2400 bytes*
CUNormal+15k-400k CU*
Latency, tx size, and CU depend on the number and type of cold accounts.

When does a market go cold?

Accounts become “cold” after extended inactivity, causing their virtual rent balance to fall below a threshold. Once cold, they auto-compress onto the Solana ledger. They remain cold until the first client loads them back into the hot state in-flight via create_load_instructions. In practice, touching cold markets is rare. The common path (“hot”) has no extra latency, tx size, or CU overhead.

Error Handling

use light_client::error::LightClientError;

match rpc.get_account_interface(&pool_address, &program_id).await {
    Ok(account) => {
        // Account is hot or cold
        // Proceed with quote and swap
    }
    Err(LightClientError::AccountNotFound) => {
        // Account does not exist
    }
}

FAQ

No. In all cases, swap instructions stay the same. If the market is active (hot), the transaction is identical to today (UX, CU, latency, txn size,…). If the market is inactive (cold), you additionally prepend create_load_instructions.
Yes. get_account_interface is a superset of get_account and returns the full account state via the same Account type, regardless of whether the account is hot or cold. Quoting works all the same.
Active markets (hot path): No additional latency.Inactive markets (cold): Yes, loading accounts back into Solana’s active state adds 1-200ms depending on whether a validity proof is needed. If loading multiple cold accounts exceeds Solana’s 1232 byte limit, use Jito bundles to maintain atomicity and reduce latency. Future updates will continue to reduce transaction size and CU usage for loading cold accounts.
Accounts stay hot until they become inactive again. After extended inactivity (configurable by the program owner, e.g., 24h of no writes), their virtual rent balance falls below a threshold and miners compress them back to cold state. Each write extends the “hot” period.
In some cases, yes. You can detect this at runtime by inspecting the Instructions returned by create_load_instructions.Note that the SDK deduplicates many of the account keys over the wire, so instructions that may appear large in isolation will be incremental when combined with other instructions, such as Swap and Deposit.If load instructions + swap instructions exceed Solana’s 1232 byte limit, send as a Jito bundle:
use solana_sdk::{instruction::Instruction, pubkey::Pubkey, system_instruction};

const JITO_TIP_ACCOUNTS: &[&str] = &[
    "96gYZGLnJYVFmbjzopPSU6QiEV5fGqZNyN9nmNhvrZU5",
    "HFqU5x63VTqvQss8hp11i4wVV8bD44PvwucfZ2bU7gRe",
    "Cw8CFyM9FkoMi7K7Crf6HNQqf4uEMzpKw6QNghXLvLkY",
    "ADaUMid9yfUytqMBgopwjb2DTLSokTSzL1zt6iGPaS49",
    "DfXygSm4jCyNCybVYYK6DwvWqjKee8pbDmJGcLWNDXjh",
    "ADuUkR4vqLUMWXxW9gh6D6L8pMSawimctcNZ5pGwDcEt",
    "DttWaMuVvTiduZRnguLF7jNxTgiMBZ1hyAumKUiL2KRL",
    "3AVi9Tg9Uo68tJfuvoKvqKNWKkC5wPdSSdeBnizKZ6jT",
];

fn jito_tip_ix(payer: &Pubkey, tip_lamports: u64) -> Instruction {
    let tip_account = JITO_TIP_ACCOUNTS[rand::random::<usize>() % JITO_TIP_ACCOUNTS.len()]
        .parse::<Pubkey>().unwrap();
    system_instruction::transfer(payer, &tip_account, tip_lamports)
}

// Add tip to last transaction, serialize, send to Jito
let tip_ix = jito_tip_ix(&payer.pubkey(), 10_000); // 10k lamports
swap_ixs.push(tip_ix);

let bundle = vec![load_tx_base64, swap_tx_base64];
let resp = client
    .post("https://mainnet.block-engine.jito.wtf/api/v1/bundles")
    .json(&serde_json::json!({
        "jsonrpc": "2.0",
        "id": 1,
        "method": "sendBundle",
        "params": [bundle, {"encoding": "base64"}]
    }))
    .send().await?;
Yes. The relevant RPC methods are supported by providers such as Helius and Triton and can also be self-hosted via the open-source Photon indexer, which is maintained by Helius Labs.
Hot markets work all the same as long as Solana is up. Cold accounts cannot be loaded into hot state until your indexer or RPC provider recovers. Note that compression is cryptographically verifiable, so integrity and safety are not dependent on the indexer or any other external service beyond the onchain protocol.

API is in Beta and subject to change.Questions or need hands-on support? Telegram | email | Discord