Back to Nautilus Trader

Lighter

docs/integrations/lighter.md

1.228.042.5 KB
Original Source

Lighter

Lighter is a decentralized central-limit-order-book exchange for spot and perpetual futures. The venue settles through an Ethereum zero-knowledge rollup, while matching and sequencing run off-chain.

The NautilusTrader Lighter adapter is implemented by the nautilus-lighter crate. It provides Rust data and execution clients, typed REST and WebSocket models, and an in-tree L2 transaction signer for the venue's Schnorr / ECgFp5 signing flow.

Overview

The adapter consists of the following main components:

  • LighterRawHttpClient: low-level REST client for the public and account endpoints.
  • LighterHttpClient: domain client which parses instruments, trades, books, orders, and account state into Nautilus model types.
  • LighterWebSocketClient: reconnecting WebSocket client for public market and private account streams.
  • LighterDataClient: Nautilus data client for instruments, trades, quotes, and L2 MBP books.
  • LighterExecutionClient: Nautilus execution client for account streams, order submission, modification, cancellation, and reconciliation reports.
  • LighterDataClientFactory and LighterExecutionClientFactory: live-node factory wiring.

The Python surface is intentionally narrow. The Python extension exposes configuration, environment selection, factory classes, and integrator revocation; data and execution clients are consumed through the Rust trait surface.

Examples

The adapter includes Python v2 and Rust live-node examples. The Python examples live in python/examples/lighter/ and default to a dry build: they build the node, register the tester, and exit unless --run is passed.

fish
cd python
.venv/bin/python examples/lighter/data_tester.py --lighter-environment testnet
.venv/bin/python examples/lighter/exec_tester.py --lighter-environment testnet

Pass --run to connect to Lighter. The execution tester remains in dry_run mode unless --live-orders is also passed.

fish
cd python
.venv/bin/python examples/lighter/data_tester.py \
    --lighter-environment mainnet \
    --instrument BTC-PERP.LIGHTER \
    --run
.venv/bin/python examples/lighter/exec_tester.py \
    --lighter-environment mainnet \
    --instrument DOGE-PERP.LIGHTER \
    --run

Rust examples live under crates/adapters/lighter/examples/ and run immediately:

fish
cargo run --example lighter-data-tester --package nautilus-lighter --features examples
cargo run --example lighter-exec-tester --package nautilus-lighter --features examples

:::warning Examples can connect to live venues. Execution examples with live order flow enabled can submit orders when pointed at a funded mainnet account. Review the selected instrument, quantity, and environment before running them. :::

Product support

Product typeData feedTradingNotes
SpotSpot markets using Lighter market indexes 2048-4094.
Perpetual futuresLinear perpetual markets using Lighter market indexes 0-254.
Dated futures--Not supported.
Options--Not supported.

Limitations

The current adapter scope is deliberately narrower than the venue's full transaction surface:

  • Grouped order lists, OCO/OTO groups, bracket orders, TWAP, trailing stops, and iceberg display size are not implemented.
  • Native batch submit and native batch cancel are wired for independent order operations only. Batch submit sends independent L2CreateOrder txs in one sendTxBatch, and batch cancel signs independent L2CancelOrder txs in one sendTxBatch. Both are capped at 15 txs per batch.
  • Grouped venue orders remain out of scope: batch submit does not use CreateGroupedOrders and does not provide atomic OCO/OTO or bracket grouping.
  • CancelAllOrders uses cached open orders for the requested instrument. The adapter does not use Lighter's native account-wide cancel-all transaction because it can affect unrelated markets.
  • Spot trading supports market and limit orders. Conditional stop-loss and take-profit orders are limited to perpetual markets.
  • Account state and position reports come from private WebSocket streams. query_account and position status generation replay the latest cached stream state.
  • Unscoped order reconciliation is bounded to configured or observed active markets to avoid a full venue-wide fan-out under the standard REST quota.
  • Historical trade requests require credentials for master accounts and sub-accounts.

Symbology

Lighter identifies markets by numeric market_index values. The adapter bootstraps the mapping from GET /api/v1/orderBookDetails, then converts the raw venue symbol into a Nautilus InstrumentId.

Venue productNautilus symbol formatExampleNotes
Perpetual futures{BASE}-PERP.LIGHTERBTC-PERP.LIGHTERRaw venue symbol BTC.
Spot{BASE}-SPOT.LIGHTERETH-SPOT.LIGHTERRaw venue symbol ETH.

The suffix disambiguates spot and perpetual listings that share the same venue symbol. Outbound requests strip the suffix and use the cached market_index.

Environments

EnvironmentREST URLWebSocket URLChain ID
Mainnethttps://mainnet.zklighter.elliot.aiwss://mainnet.zklighter.elliot.ai/stream304
Testnethttps://testnet.zklighter.elliot.aiwss://testnet.zklighter.elliot.ai/stream300

Use LighterEnvironment::Mainnet or LighterEnvironment::Testnet in data and execution configuration. URL overrides are available for private gateways or local test fixtures.

Integrator attribution

Submitted create and modify order transactions carry the NautilusTrader integrator account index in Lighter's L2TxAttributes. This helps us gauge real usage of the integration and prioritize ongoing maintenance. Maker and taker integrator fees are set to zero, so attribution adds no trading cost.

Lighter requires an ApproveIntegrator approval before these attributes can be attached to orders. During startup, the execution client submits the required zero-fee approval for the configured L2 account.

Revoking the approval

Use revocation as cleanup when leaving the adapter. It sends the same ApproveIntegrator tx with approval_expiry = 0 and every max fee set to zero; the next execution-client startup records a fresh zero-fee approval.

bash
export LIGHTER_API_KEY_INDEX=0
export LIGHTER_API_SECRET=REPLACE_ME
export LIGHTER_ACCOUNT_INDEX=123456
cargo run -p nautilus-lighter --bin lighter-integrator-revoke           # mainnet
cargo run -p nautilus-lighter --bin lighter-integrator-revoke testnet   # testnet

Script source: crates/adapters/lighter/bin/integrator_revoke.rs.

python
# Python (PyO3 binding) - reads the same env vars as the Rust bin
from nautilus_trader.core.nautilus_pyo3 import revoke_lighter_integrator
from nautilus_trader.core.nautilus_pyo3 import LighterEnvironment

await revoke_lighter_integrator()                            # mainnet (default)
await revoke_lighter_integrator(LighterEnvironment.TESTNET)  # testnet

The Rust script prints a summary of the action and pauses for an Enter keypress before signing or sending; abort with Ctrl+C before that point if anything in the summary looks wrong. The Python binding does not prompt: review the active env vars yourself before calling.

Data subscriptions

Data typeSub.SnapshotHist.Nautilus typeNotes
Instrument metadataCache replay-InstrumentAnyLoaded from orderBookDetails.
Trade ticks-TradeTickWebSocket trades; historical REST trades require auth.
Quote ticks--QuoteTickBest bid and ask ticker stream.
Order book deltas-OrderBookDeltasL2_MBP only.
Order book depth10-OrderBookDepth10Full WebSocket book snapshots.
Order book snapshots--OrderBookREST snapshot, max depth 250.
Mark prices--MarkPriceUpdatePerp market stats stream.
Index prices--IndexPriceUpdateMarket and spot stats streams.
Funding rates-FundingRateUpdateCurrent estimates and REST hourly history.
Bars-BarWebSocket candle stream; REST history for backfill.
Instrument statusREST-InstrumentStatusactive / inactive snapshots.

Only BookType::L2_MBP is accepted for book-delta and depth10 subscriptions. Other book types return an error before subscribing.

The WebSocket order book initializes only from subscribed/order_book. If an update/order_book arrives before that snapshot, the adapter drops it and waits for the real snapshot because incremental updates do not contain the full visible book.

Bar subscriptions use the venue's candle/{market_id}/{resolution} WebSocket channel. Lighter batches in-progress updates for the open bar every ~500 ms; the adapter emits a Nautilus Bar only when the candle start timestamp advances, so consumers see one event per closed period. The in-progress cache is cleared on reconnect and on unsubscribe.

The stream supports 1m, 5m, 15m, 30m, 1h, 4h, 12h, and 1d. 1w is REST-only via request_bars; subscribing to a 1-WEEK bar type returns an error.

Instrument status subscriptions replay the latest cached orderBookDetails status when available and otherwise fetch a REST snapshot. Lighter does not expose a WebSocket status-change stream.

Funding-rate subscriptions use market_stats.current_funding_rate, which is Lighter's estimate for the upcoming payment. Historical funding-rate requests use /api/v1/fundings at 1h resolution and map settled rows to FundingRateUpdate with interval=60. The REST direction field controls the sign: long stays positive because longs pay shorts, while short is mapped negative because shorts pay longs. The adapter does not use account-specific positionFunding payloads for public funding history.

Trade subscriptions use the public WebSocket trade stream. Historical trade requests use /api/v1/trades; live mainnet testing shows that endpoint rejects unauthenticated requests. The data client mints a Lighter auth token for this request when credentials are available. Without credentials, the data client logs a warning and rejects the request.

Unsupported data requests

request_quotes is not implemented. Lighter exposes best bid and offer data through the WebSocket ticker stream, but the REST endpoints available to the adapter do not provide a timestamped quote snapshot or quote history that can map safely to QuoteTick.

request_book_depth is not implemented. The documented REST book endpoints do not provide a venue event timestamp for OrderBookDepth10.ts_event; use subscribe_book_depth10 for live depth10 snapshots or request_book_snapshot for a REST OrderBook snapshot.

Orders capability

Order identification

Lighter uses a numeric venue order index and a caller-supplied client_order_index. The adapter derives the Lighter client_order_index from the Nautilus ClientOrderId and keeps a local map so private WebSocket reports can recover the original client order ID.

Query paths can use either the Nautilus client order ID or the numeric venue order ID when the required mapping is available.

Order types

Order typePerpetualsSpotNotes
MARKETCap derived from cached far‑side quote + slippage.
LIMITRequires a limit price.
STOP_MARKET-Perp only; cap derived from trigger_price + slippage.
STOP_LIMIT-Perp only; maps to Lighter stop‑loss limit orders.
MARKET_IF_TOUCHED-Perp only; cap derived from trigger_price + slippage.
LIMIT_IF_TOUCHED-Perp only; maps to Lighter take‑profit limit orders.
MARKET_TO_LIMIT--Not supported.
TRAILING_STOP_MARKET--Not supported.
TRAILING_STOP_LIMIT--Not supported.
TWAP--Not supported; no Nautilus mapping.

Conditional order types are available for perpetual markets only. Spot conditional orders are denied locally because Lighter rejects them at the venue. Conditional order types must include a trigger_price. STOP_MARKET and MARKET_IF_TOUCHED are denied upfront if the trigger is missing, and all conditional types are denied if the trigger truncates to 0 ticks at the instrument's price precision.

Lighter's market-style orders require a worst-acceptable price field on the wire. The adapter derives it automatically: MARKET orders read the cached far-side QuoteTick (ask for buys, bid for sells); STOP_MARKET and MARKET_IF_TOUCHED use the order's trigger_price. The base is widened by market_order_slippage_bps (default 50 bps = 0.5%), rounded conservatively at the instrument's price precision (ceil for buys, floor for sells). A MARKET order submitted before the strategy has subscribed to quotes is denied with a clear error. Override per order via SubmitOrder.params["market_order_slippage_bps"].

Contingent orders

FeaturePerpetualsSpotNotes
Stop‑loss market-STOP_MARKET maps to Lighter STOP_LOSS.
Stop‑loss limit-STOP_LIMIT maps to Lighter STOP_LOSS_LIMIT.
Take‑profit market-MARKET_IF_TOUCHED maps to Lighter TAKE_PROFIT.
Take‑profit limit-LIMIT_IF_TOUCHED maps to TAKE_PROFIT_LIMIT.
Trigger price-Required for every supported conditional order.
Trigger price type--Not supported; no trigger source selector.
Grouped order lists--Not supported.
OCO / OTO orders--Not supported.
Bracket orders--Not supported.
CreateGroupedOrders--Not supported; native batches use independent txs.

Order options

OptionPerpetualsSpotNotes
post_onlyMaps to Lighter's post‑only time‑in‑force.
reduce_only-Passed through to CreateOrder; use only to reduce an existing position.
quote_quantity--Not supported; submit base quantity instead.
display_qty--Not supported; Lighter exposes no iceberg display quantity field.

Adapter order params

ParamPerpetualsSpotNotes
market_order_slippage_bpsOverrides the config default for market‑style caps.
post_only through SubmitOrder.params--Not supported; use the Nautilus order flag.
reduce_only through SubmitOrder.params--Not supported; use the Nautilus order flag.

Time in force

Time in forcePerpetualsSpotNotes
GTCLimit‑style uses GoodTillTime; market‑style uses IOC.
DAYLimit‑style and conditional orders use a positive order expiry.
GTDLimit‑style and conditional orders use the supplied Nautilus expiry.
IOCPlain MARKET/LIMIT use expiry 0; conditional limit uses trigger expiry.
FOK--Not supported.
AT_THE_OPEN--Not supported.
AT_THE_CLOSE--Not supported.

For MARKET, STOP_MARKET, and MARKET_IF_TOUCHED, the adapter maps the wire time-in-force to Lighter ImmediateOrCancel because the venue rejects market-style orders sent as GoodTillTime. Plain MARKET orders set OrderExpiry = 0. Conditional market orders (STOP_MARKET and MARKET_IF_TOUCHED) keep a positive OrderExpiry so the trigger can rest, and the wire ImmediateOrCancel applies only after the trigger fires. Nautilus IOC cannot be represented for conditional market orders, so the adapter denies it locally with a clear error. Conditional limit orders (STOP_LIMIT and LIMIT_IF_TOUCHED) can use Nautilus IOC: the trigger rests with a positive OrderExpiry, and the child limit order uses Lighter ImmediateOrCancel after the trigger fires.

When no explicit GTD expiry is supplied, limit-style GTC, DAY, and GTD orders default to the current time plus 28 days. Conditional GTC, DAY, and limit-style IOC orders use the same default expiry. The venue rejects -1 as an invalid expiry for these TIFs. Live testing has also shown that very short GTD expiries can be rejected by the sequencer with 21711 invalid expiry; use a venue-accepted expiry horizon for live GTD tests.

Execution instructions

InstructionPerpetualsSpotNotes
post_onlyOverrides the TIF and sends Lighter PostOnly.
reduce_only-Position‑reducing flag for existing derivative positions.

Use post_only on limit-style orders. The adapter does not synthesize maker-only market orders. Live mainnet testing confirms reduce_only=true for closing perpetual positions. Invalid reduce-only opens can be dropped by Lighter without a venue order report; the adapter reconciles them as INFLIGHT_TIMEOUT rather than a venue-supplied rejection reason.

Advanced order features

FeaturePerpetualsSpotNotes
Order modificationModify quantity, price, and trigger price on a live order.
Bracket orders--Not supported.
Iceberg orders--Not supported.
Trailing stops--Not supported.
Pegged orders--Not supported.
TWAP orders--Not supported; no Nautilus mapping.
Leverage update-Perp only; submits a signed UpdateLeverage tx.
Native cancel‑all--Not supported; adapter scopes cancel‑all per instrument.
Dead man's switch--Not supported.

Order operations

OperationPerpetualsSpotNotes
Submit orderSends a signed L2CreateOrder transaction over WebSocket.
Submit order listBatches independent L2CreateOrder txs only.
Modify orderSends a signed ModifyOrder; reports may restate accepts.
Cancel orderSends a signed L2CancelOrder transaction.
Cancel all ordersIterates cached open orders for the requested instrument.
Set leverage-Perp only; submits a signed UpdateLeverage tx.
Batch cancel ordersBatches independent L2CancelOrder txs only.
Native batch submitUses one sendTxBatch, capped at 15 create txs.
Native batch cancelUses one sendTxBatch, capped at 15 cancel txs.
Query orderRequires credentials and REST lookup.
Query accountReplays the latest private WebSocket account state.
Mass statusBounded to account‑active markets from WS and REST reports.

The native venue CancelAllOrders transaction is account-wide. The adapter deliberately cancels cached open orders per instrument to avoid touching unrelated markets.

SubmitOrderList and BatchCancelOrders use sendTxBatch for independent operations. They do not create grouped venue orders, do not provide atomic OCO/OTO or bracket semantics, and do not use account-wide CancelAllOrders for scoped cancels.

The sendTxBatch response exposes one top-level API code and a tx_hash list; it does not expose per-order API rejection fields. A successful batch response queues the signed transactions, then private account streams report the final per-order submit, cancel, fill, and reject outcomes.

UpdateLeverage is exposed as LighterExecutionClient::update_leverage(instrument_id, initial_margin_fraction, margin_mode). The initial_margin_fraction is in venue ticks (1e-4 fraction): 500 is 5% initial margin (20x leverage), 1000 is 10% (10x), and so on.

UpdateLeverage has no oracle test vectors in this repo; the body field order is pinned against the cgo header from the upstream signer, and the wire format was verified by submitting a signed tx to Lighter mainnet that the sequencer accepted.

Order querying and reconciliation

FeaturePerpetualsSpotNotes
Query open ordersREST accountActiveOrders scoped by market.
Query order historyREST accountInactiveOrders with cursor pagination.
Order status updatesPrivate WebSocket order streams plus status reports.
Trade historyREST trades; credentials are required for account history.
Fill reportsREST and private WebSocket trade payloads.
Position reports-Perp only; replays cached position stream.
Account stateReplays the cached account_all_assets stream.
Mass statusCombines orders, fills, and cached positions.

Account and position management

Authenticated execution clients subscribe to these private streams:

  • account_all_orders: order status reports.
  • account_all_trades: fill reports.
  • account_all_positions: position snapshots.
  • account_all_assets: account balance and margin snapshots.

The execution client requires credentials before connecting because private account streams and nonce refresh are mandatory. A client can be constructed without credentials, but live execution will not connect until private_key, account_index, and api_key_index resolve.

Perpetual positions are reported in netting mode: one position per market. Spot balances arrive through account asset state rather than position reports. Each account_all_positions frame is treated as a complete venue snapshot. If a new frame omits a previously cached market, the adapter emits a flat position report for that instrument; an empty positions map clears all cached perpetual positions and emits a flat report for each.

FeaturePerpetualsSpotNotes
Account balancesaccount_all_assets stream, replayed from cache on query.
Position snapshots-Perp only; account_all_positions stream.
Netting positions-One Nautilus position per perpetual market.
Cross margin-Passed through LighterPositionMarginMode::Cross.
Isolated margin-Passed through LighterPositionMarginMode::Isolated.
Leverage updates-Signed UpdateLeverage transaction.
Spot margin / borrowing--Not supported.
Deposits / withdrawals--Use venue tools or Lighter APIs outside the trading adapter.

Liquidation and ADL handling

Event or fieldSupportNotes
Liquidation tradesAccount trade rows can parse as fills, with no special event.
Deleverage tradesAccount trade rows can parse as fills, with no special event.
Liquidation price reporting-Not supported; reports omit this field.
ADL event stream-Not supported.

Funding rates

Perpetual market_stats frames emit MarkPriceUpdate, IndexPriceUpdate, and FundingRateUpdate events. Spot spot_market_stats frames emit IndexPriceUpdate events.

Historical funding-rate requests use the public /api/v1/fundings endpoint and emit FundingRateUpdate responses for settled hourly rows.

Account tiers

Lighter assigns each account a tier that governs latency, rate limits, and fees. Standard is the zero-fee default; the higher tiers are opt-in on the venue and trade fees for lower latency and higher throughput. The execution client detects the tier on connect (via GET /api/v1/account) and logs it in blue, including the raw account_type code for any tier this adapter does not yet recognize. Detection is informational only: the adapter never raises rate limits on its own, because the higher venue limits require registering the caller IP with Lighter, so a higher tier does not by itself guarantee the higher limit is active for your connection.

TierLatency (maker / taker)REST weighted limitsendTx limitFees (maker / taker)Notes
Standard200 ms / 300 ms60 req/min60 req/min0 / 0Zero‑fee default tier.
Premium0 ms / 140-200 ms24,000 req/min4,000-40,000 req/min0.28-0.40 / 1.96-2.80 bpsLowest latency; scales with staked LIT.
Plus200 ms / 300 ms120,000 req/min8,000 req/min0.5 / 0.5 bpsRaised limits, standard latency.
Builder-240,000 req/min--Highest REST throughput.

Premium latency, fees, and sendTx throughput scale with staked LIT, and the schedule can change; see the Lighter docs for the current figures. To actually use a higher tier's limits, register the caller IP with Lighter and set the quota explicitly (see Rate limiting).

Rate limiting

Lighter applies rate limits to both IP address and L1 address. The execution client detects the account tier on connect and logs it (see Account tiers), but it does not raise limits automatically. By default both clients use the conservative standard-account quotas. To use a higher tier's throughput, register the caller IP with Lighter and set the quota explicitly in the client config:

  • rest_quota_per_min: REST read-bucket quota in requests per minute. Unset keeps 60 req/min. Available on both the data and execution clients.
  • sendtx_quota_per_min: transaction quota in requests per minute, metered in a bucket separate from reads. Unset keeps it at the standard 60 req/min, independent of rest_quota_per_min. Execution client only.

The venue meters transactions per account across both transports in one bucket. The execution client enforces sendtx_quota_per_min with a single shared limiter across both paths it submits on: the WebSocket sendTx path (single order submit, cancel, modify, leverage) and the HTTP sendTx / sendTxBatch endpoints (native batch submit/cancel and the startup integrator approval). Their combined rate therefore stays under the one venue limit.

ScopeVenue limitAdapter behavior
REST, standard account60 req/minDefault; set rest_quota_per_min to override.
REST, premium account24,000 weighted req/minLogged; set rest_quota_per_min to use it.
REST, plus account120,000 weighted req/minLogged; set rest_quota_per_min to use it.
REST, builder account240,000 weighted req/minLogged; set rest_quota_per_min to use it.
sendTx / sendTxBatch, standard60 req/minSingles use sendTx; batches use sendTxBatch.
sendTx / sendTxBatch, plus8,000 req/minSet sendtx_quota_per_min to use it.
sendTx / sendTxBatch, premium4,000-40,000 req/minSet sendtx_quota_per_min (scales with staked LIT).
Default transaction type limit40 req/minApplies to tx types not covered by volume quota.
L2UpdateLeverage transaction limit40 req/minRelevant to update_leverage.
Pending orders500/account, 16/marketVenue limit; adapter does not pre‑count it.
Active orders1,500/account, 1,000/marketVenue limit; adapter does not pre‑count it.
Endpoint or transportLimitNotes
/api/v1/trades100 rowsAdapter paginates reconciliation at this cap.
/api/v1/accountInactiveOrders100 rowsAdapter follows next_cursor at this cap.
/api/v1/orderBookOrders250 levelsSnapshot depth is clamped to the venue cap.
/api/v1/candles500 rowsAdapter caps REST bar pages at this venue maximum.
WebSocket connections200 / IPVenue limit.
WebSocket subscriptions / connection500Venue limit.
WebSocket unique accounts / connection500Venue limit.
WebSocket connections / minute80Venue limit.
WebSocket client messages / minute200Excludes sendTx and sendTxBatch.
WebSocket inflight messages50Excludes sendTx and sendTxBatch.
sendTxBatch batch size15 txsApplies to native HTTP submit and cancel batches.
WebSocket keepalive2 minutesAdapter sends heartbeats every 30 seconds.
WebSocket outbound command queue1000Adapter backpressure starts at this queue depth.

Premium volume quota is a separate venue constraint for L2CreateOrder, L2CancelAllOrders, L2ModifyOrder, and L2CreateGroupedOrders. The adapter does not inspect remaining quota; use venue account tools if a strategy depends on premium or plus limits.

Connection management

The WebSocket client sends heartbeats every 30 seconds and reconnects with exponential backoff from 250 milliseconds up to 30 seconds. Private account subscriptions use Lighter auth tokens with an 8-hour maximum TTL; the adapter refreshes tokens 15 minutes before expiry and resubscribes account channels.

On execution reconnect, the adapter refreshes the nonce baseline through GET /api/v1/nextNonce before it resumes signed transaction dispatch.

LighterExecutionClient::connect() waits up to 30 seconds for every account stream (account_all_orders, account_all_trades, account_all_positions, account_all_assets) to deliver its first frame before returning. Lighter has no REST endpoint for account or position state, so the WebSocket frames are the only ground truth: returning earlier would let strategies race the venue's initial state and find the venue order id lookup table or position cache empty. The gate clears any prior-session position and account caches at the start of each connect attempt so a reconnect cycle observes the new session's frames, not stale data. By contrast, transparent WebSocket reconnects do not re-enter connect(): they keep cached positions until the next account_all_positions frame, then apply the same complete-snapshot replacement rules.

API credentials

Lighter signing requires all three credential values:

  • Account index: numeric Lighter account identifier.
  • API key index: numeric API key slot, 0..=254. Indices 0..=3 are reserved for Lighter desktop/mobile clients.
  • API private key: 40-byte hex private key, with or without a 0x prefix.

Config values take precedence. When config fields are omitted, the adapter reads environment variables based on the selected environment.

EnvironmentAPI key indexAPI private keyAccount index
MainnetLIGHTER_API_KEY_INDEXLIGHTER_API_SECRETLIGHTER_ACCOUNT_INDEX
TestnetLIGHTER_TESTNET_API_KEY_INDEXLIGHTER_TESTNET_API_SECRETLIGHTER_TESTNET_ACCOUNT_INDEX

Execution rejects incomplete credentials. The data client can run without credentials for public streams and public REST endpoints; authenticated data requests such as request_trades use the same values when all three are available.

Configuration

Data client configuration options

OptionDefaultDescription
base_url_httpNoneOptional REST URL override.
base_url_wsNoneOptional WebSocket URL override.
proxy_urlNoneOptional proxy URL for HTTP and WebSocket.
environmentMainnetLighterEnvironment::Mainnet or Testnet.
account_indexNoneLighter account index for authenticated REST data.
api_key_indexNoneLighter API key slot for authenticated REST data.
private_keyNoneHex private key for REST auth tokens.
http_timeout_secs60HTTP request timeout in seconds.
ws_timeout_secs30WebSocket connect timeout in seconds.
update_instruments_interval_mins60Instrument metadata refresh interval in minutes.
transport_backendDefaultWebSocket transport backend.

Execution client configuration options

OptionDefaultDescription
trader_idRequiredNautilus trader identifier.
account_idRequiredNautilus account identifier for the venue.
account_indexNoneLighter account index.
api_key_indexNoneLighter API key slot.
private_keyNoneHex private key for auth and L2 transaction signing.
base_url_httpNoneOptional REST URL override.
base_url_wsNoneOptional WebSocket URL override.
proxy_urlNoneOptional proxy URL for HTTP and WebSocket.
environmentMainnetLighterEnvironment::Mainnet or Testnet.
http_timeout_secs60HTTP request timeout in seconds.
ws_timeout_secs30WebSocket connect timeout in seconds.
active_markets[]Lighter market IDs to poll during unscoped reconciliation.
market_order_slippage_bps50Slippage cap (bps) for MARKET / STOP_MARKET / MIT.
transport_backendDefaultWebSocket transport backend.

Configuration example

rust
use nautilus_lighter::{
    common::enums::LighterEnvironment,
    config::{LighterDataClientConfig, LighterExecClientConfig},
};

let data_config = LighterDataClientConfig {
    environment: LighterEnvironment::Testnet,
    ..Default::default()
};

let exec_config = LighterExecClientConfig::builder()
    .trader_id(trader_id)
    .account_id(account_id)
    .environment(LighterEnvironment::Testnet)
    .active_markets(vec![0])
    .build();

The execution config above resolves credentials from the matching testnet environment variables. Set account_index, api_key_index, and private_key directly to override environment lookup. Set active_markets to the venue market IDs that should be checked for open orders during cold-start reconciliation.

Official documentation

Contributing

:::info For additional features or to contribute to the Lighter adapter, please see our contributing guide. :::