- Closing a Light Token account transfers remaining lamports to a destination account and the rent sponsor can reclaim sponsored rent.
- Light token accounts can be closed by the owner.
The
closes the account and preserves the balance as compressed token account when the account becomes .
The account is reinstated in flight with the same state the next time it is accessed.
- Rust Client
- Program
Use
CloseTokenAccount to close an empty Light Token account.Compare to SPL:Prerequisites
Dependencies
Dependencies
Cargo.toml
Report incorrect code
Copy
Ask AI
[dependencies]
light-token = "0.4.0"
light-client = { version = "0.19.0", features = ["v2"] }
solana-sdk = "2"
borsh = "0.10.4"
tokio = { version = "1", features = ["full"] }
Developer Environment
Developer Environment
- In-Memory (LightProgramTest)
- Localnet (LightClient)
- Devnet (LightClient)
Test with Lite-SVM (…)
Report incorrect code
Copy
Ask AI
# Initialize project
cargo init my-light-project
cd my-light-project
# Run tests
cargo test
Report incorrect code
Copy
Ask AI
use light_program_test::{LightProgramTest, ProgramTestConfig};
use solana_sdk::signer::Signer;
#[tokio::test]
async fn test_example() {
// In-memory test environment
let mut rpc = LightProgramTest::new(ProgramTestConfig::default())
.await
.unwrap();
let payer = rpc.get_payer().insecure_clone();
println!("Payer: {}", payer.pubkey());
}
Connects to a local test validator.
- npm
- yarn
- pnpm
Report incorrect code
Copy
Ask AI
npm install -g @lightprotocol/zk-compression-cli@beta
Report incorrect code
Copy
Ask AI
yarn global add @lightprotocol/zk-compression-cli@beta
Report incorrect code
Copy
Ask AI
pnpm add -g @lightprotocol/zk-compression-cli@beta
Report incorrect code
Copy
Ask AI
# Initialize project
cargo init my-light-project
cd my-light-project
# Start local test validator (in separate terminal)
light test-validator
Report incorrect code
Copy
Ask AI
use light_client::rpc::{LightClient, LightClientConfig, Rpc};
#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
// Connects to http://localhost:8899
let rpc = LightClient::new(LightClientConfig::local()).await?;
let slot = rpc.get_slot().await?;
println!("Current slot: {}", slot);
Ok(())
}
Replace
<your-api-key> with your actual API key. Get your API key here.Report incorrect code
Copy
Ask AI
use light_client::rpc::{LightClient, LightClientConfig, Rpc};
#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
let rpc_url = "https://devnet.helius-rpc.com?api-key=<your_api_key>";
let rpc = LightClient::new(
LightClientConfig::new(rpc_url.to_string(), None, None)
).await?;
println!("Connected to Devnet");
Ok(())
}
Close Light Token Account
View the source code and full example with shared test utilities.
- Instruction
Report incorrect code
Copy
Ask AI
use light_client::rpc::Rpc;
use light_token::instruction::{CloseAccount, LIGHT_TOKEN_PROGRAM_ID};
use rust_client::{setup_empty_associated_token_account, SetupContext};
use solana_sdk::signer::Signer;
#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
// Setup creates mint and empty associated token account (must be empty to close).
let SetupContext {
mut rpc,
payer,
associated_token_account,
..
} = setup_empty_associated_token_account().await;
let close_instruction = CloseAccount::new(LIGHT_TOKEN_PROGRAM_ID, associated_token_account, payer.pubkey(), payer.pubkey())
.instruction()?;
let sig = rpc
.create_and_send_transaction(&[close_instruction], &payer.pubkey(), &[&payer])
.await?;
let account = rpc.get_account(associated_token_account).await?;
println!("Closed: {} Tx: {sig}", account.is_none());
Ok(())
}
Build Account Infos and CPI the Light Token Program
Useinvoke for external signers or invoke_signed when the authority is a PDA.- invoke (External signer)
- invoke_signed (PDA owner)
Report incorrect code
Copy
Ask AI
use light_token::instruction::CloseAccountCpi;
CloseAccountCpi {
token_program: token_program.clone(),
account: account.clone(),
destination: destination.clone(),
owner: owner.clone(),
rent_sponsor: rent_sponsor.clone(),
}
.invoke()?;
Report incorrect code
Copy
Ask AI
use light_token::instruction::CloseAccountCpi;
CloseAccountCpi {
token_program: token_program.clone(),
account: account.clone(),
destination: destination.clone(),
owner: owner.clone(),
rent_sponsor: rent_sponsor.clone(),
}
.invoke_signed(&[signer_seeds])?;
Account List
Account List
| - | The light token program for CPI. | |
| mutable | The light-token account to close. | |
| mutable | Receives remaining lamports from the closed account. | |
| signer* |
| |
| mutable, optional |
|
Full Code Example
View the source code and full example with shared test utilities.
Report incorrect code
Copy
Ask AI
#![allow(unexpected_cfgs, deprecated)]
use anchor_lang::prelude::*;
use light_token::instruction::CloseAccountCpi;
declare_id!("GXLCuNhnkRVp596eCdbNsZ9ua1ePbKbb344VKS7V3zQQ");
#[program]
pub mod light_token_anchor_close {
use super::*;
pub fn close_account(ctx: Context<CloseAccountAccounts>) -> Result<()> {
CloseAccountCpi {
token_program: ctx.accounts.light_token_program.to_account_info(),
account: ctx.accounts.account.to_account_info(),
destination: ctx.accounts.destination.to_account_info(),
owner: ctx.accounts.owner.to_account_info(),
rent_sponsor: ctx.accounts.rent_sponsor.to_account_info(),
}
.invoke()?;
Ok(())
}
}
#[derive(Accounts)]
pub struct CloseAccountAccounts<'info> {
/// CHECK: Light token program for CPI
pub light_token_program: AccountInfo<'info>,
/// CHECK: Validated by light-token CPI
#[account(mut)]
pub account: AccountInfo<'info>,
/// CHECK: Validated by light-token CPI
#[account(mut)]
pub destination: AccountInfo<'info>,
pub owner: Signer<'info>,
/// CHECK: Validated by light-token CPI
#[account(mut)]
pub rent_sponsor: AccountInfo<'info>,
}