lasersell-sdk 1.1.0

Rust SDK for the LaserSell API
Documentation

lasersell-sdk

Rust SDK for the LaserSell API.

Full documentation: docs.lasersell.io/api/sdk/rust

Modules

Using AI to code? Add the LaserSell MCP server to your editor so your AI assistant can search LaserSell documentation in real time.

Install

cargo add lasersell-sdk

Build a sell transaction

use lasersell_sdk::exit_api::{BuildSellTxRequest, ExitApiClient, SellOutput};
use secrecy::SecretString;

let client = ExitApiClient::with_api_key(SecretString::new("REPLACE_WITH_API_KEY".to_string()))?;

let request = BuildSellTxRequest {
    mint: "REPLACE_WITH_MINT".to_string(),
    user_pubkey: "REPLACE_WITH_WALLET_PUBKEY".to_string(),
    amount_tokens: 1_000_000,
    output: SellOutput::Sol,
    slippage_bps: 2_000,
    ..Default::default()
};

let response = client.build_sell_tx(&request).await?;
println!("{}", response.tx);

Stream + auto-sell flow

Important: Connect the stream before submitting a buy transaction. See docs.lasersell.io/api/exit-api/buy for details.

use lasersell_sdk::stream::client::{StreamClient, StreamConfigure, strategy_config_from_optional};
use lasersell_sdk::stream::session::{StreamSession, StreamEvent};
use lasersell_sdk::stream::proto::ServerMessage;
use lasersell_sdk::tx::{sign_unsigned_tx, send_transaction, SendTarget};
use secrecy::SecretString;
use solana_sdk::signature::read_keypair_file;

let keypair = read_keypair_file("REPLACE_WITH_KEYPAIR_PATH")?;
let http = reqwest::Client::builder().no_proxy().build()?;

let client = StreamClient::new(SecretString::new("REPLACE_WITH_API_KEY".to_string()));
let configure = StreamConfigure {
    wallet_pubkeys: vec!["REPLACE_WITH_WALLET_PUBKEY".to_string()],
    strategy: strategy_config_from_optional(None, None, None, None),
    deadline_timeout_sec: 45,
    send_mode: Some("helius_sender".to_string()),
    tip_lamports: Some(1000),
};

let mut session = StreamSession::connect(&client, configure).await?;

while let Some(event) = session.recv().await {
    if let StreamEvent::ExitSignalWithTx {
        message: ServerMessage::ExitSignalWithTx { unsigned_tx_b64, .. },
        ..
    } = event
    {
        let signed_tx = sign_unsigned_tx(&unsigned_tx_b64, &keypair)?;
        let signature = send_transaction(&http, &SendTarget::HeliusSender, &signed_tx).await?;
        println!("signature={signature}");
    }
}

Strategy options

The strategy_config_from_optional helper builds a StrategyConfigMsg from optional fields. At least one exit condition must be enabled: take profit, stop loss, trailing stop, or deadline timeout.

use lasersell_sdk::stream::client::strategy_config_from_optional;

// Take profit at 20%, stop loss at 10%, trailing stop at 5%
let strategy = strategy_config_from_optional(Some(20.0), Some(10.0), Some(5.0), None);

Trailing stop locks in profits by tracking a high-water mark. When the position's profit drops by the configured percentage of entry cost from its peak, an exit is triggered. For example, with trailing_stop_pct = 5.0 and an entry of 100 SOL: if profit peaks at 30 SOL, an exit triggers when profit falls below 25 SOL.

Sell on graduation

Automatically exit a position when its token graduates from a bonding curve to a full DEX (e.g. Pump.fun to PumpSwap). Enable by setting sell_on_graduation on the strategy config:

use lasersell_sdk::stream::proto::StrategyConfigMsg;

let strategy = StrategyConfigMsg {
    target_profit_pct: 20.0,
    stop_loss_pct: 10.0,
    trailing_stop_pct: 0.0,
    sell_on_graduation: true,
};

When a graduation event is detected the server sells the position on the new market and reports "graduation" as the exit reason.

Update wallets mid-session

Add or remove wallets without reconnecting:

session.sender().update_wallets(vec![
    "WALLET_1_PUBKEY".to_string(),
    "WALLET_2_PUBKEY".to_string(),
]).await?;

The server diffs the new list against the current set and registers/unregisters accordingly.

Liquidity snapshots (Tier 1+)

Professional and Advanced tier subscribers receive real time liquidity snapshots alongside PnL updates. Each snapshot contains slippage bands and a liquidity trend indicator. See the full announcement. StreamSession caches the latest snapshot per position:

let bands = session.get_slippage_bands(position_id);
let max_tokens = session.get_max_sell_at_slippage(position_id, 500); // 5% slippage
let trend = session.get_liquidity_trend(position_id); // "growing" | "stable" | "draining"

Partial sell

Build a sell transaction for a subset of a position's tokens using the PositionHandle from any StreamEvent:

let response = client.build_partial_sell_tx(&handle, amount_tokens, 500, SellOutput::Sol).await?;

Combine with slippage bands to sell the maximum amount within your desired price impact.

Exit ladder

Sell partial amounts at multiple profit thresholds:

use lasersell_sdk::stream::client::StrategyConfigBuilder;
use lasersell_sdk::stream::proto::TakeProfitLevelMsg;

let strategy = StrategyConfigBuilder::new()
    .stop_loss_pct(10.0)
    .take_profit_levels(vec![
        TakeProfitLevelMsg { profit_pct: 25.0, sell_pct: 30.0, trailing_stop_pct: 0.0 },
        TakeProfitLevelMsg { profit_pct: 50.0, sell_pct: 50.0, trailing_stop_pct: 3.0 },
        TakeProfitLevelMsg { profit_pct: 100.0, sell_pct: 100.0, trailing_stop_pct: 5.0 },
    ])
    .build();

Liquidity guard

Prevent exits into thin liquidity:

let strategy = StrategyConfigBuilder::new()
    .target_profit_pct(50.0)
    .stop_loss_pct(10.0)
    .liquidity_guard(true)
    .build();

Breakeven trail

A trailing stop that activates at breakeven:

let strategy = StrategyConfigBuilder::new()
    .target_profit_pct(50.0)
    .stop_loss_pct(10.0)
    .breakeven_trail_pct(2.0)
    .build();

Per-position strategy override

Override the global strategy for a single position:

session.sender().update_position_strategy(position_id, StrategyConfigMsg {
    target_profit_pct: 200.0,
    stop_loss_pct: 5.0,
    trailing_stop_pct: 10.0,
    ..Default::default()
})?;

Wallet registration

All wallets must be registered before connecting to the Exit Intelligence Stream:

use lasersell_sdk::exit_api::prove_ownership;

// 1. Prove ownership (local Ed25519 signature)
let proof = prove_ownership(&keypair);

// 2. Register with the API
client.register_wallet(&proof, Some("My Wallet")).await?;

// 3. Connect with registered wallet (convenience method)
let session = stream_client.connect_with_wallets(&[proof], strategy, 120).await?;

Watch wallets (copy trading)

Mirror trades from external wallets:

session.sender().update_watch_wallets(vec![
    WatchWalletEntryMsg { pubkey: "WalletToWatch...".into(), auto_buy: None },
])?;

Priority lanes

Split the stream into high-priority and low-priority channels. Exit signals flow through the high-priority channel while PnL updates and liquidity snapshots flow through the low-priority channel:

let connection = StreamSession::connect(&client, configure).await?;
let lanes = connection.into_lanes(1024); // low-priority channel buffer size

RPC endpoint

The SDK ships with the Solana public RPC as a default so you can get started immediately:

let target = SendTarget::default_rpc();

A private RPC is highly recommended for production — the public endpoint is rate-limited and unreliable under load. Free private RPC tiers are available from Helius and Chainstack, among others:

let target = SendTarget::Rpc { url: "https://your-private-rpc.example.com".to_string() };

Examples

See examples/ for runnable programs:

  • examples/build_buy.rs: Build unsigned buy tx
  • examples/build_sell.rs: Build unsigned sell tx
  • examples/build_and_send_sell.rs: Build, sign, and submit sell tx
  • examples/auto_sell.rs: Stream listener that signs and submits exits
cargo run --example build_sell
cargo run --example build_and_send_sell
cargo run --example auto_sell

Error types

  • ExitApiError: API transport and response errors
  • TxSubmitError: Transaction signing and submission errors

Both implement std::error::Error and can be matched by variant for structured error handling.

License

MIT