lifi
LI.FI Cross-Chain Integration Guide
Use LI.FI when a Solana app, wallet, backend, or AI agent needs cross-chain token transfers, bridge aggregation, same-chain swaps, payment flows, or transfer status tracking. LI.FI exposes the same routing engine through REST API, SDK, MCP server, CLI, and Widget surfaces.
Overview
LI.FI is a multi-chain liquidity aggregation platform for swaps and bridging. For agent and backend integrations, the REST API is the lowest-dependency default. For frontend apps, prefer the SDK because it handles wallets, signing, execution tracking, and ecosystem-specific transaction handling. For MCP-compatible AI hosts, prefer the LI.FI MCP server for typed tool discovery.
Core capabilities:
- Quotes: Get a ready-to-sign transaction for the best route.
- Routes: Compare multiple route options and execute step-by-step.
- Status tracking: Track source and destination chain transfer progress.
- Discovery: Query supported chains, tokens, bridges, and exchanges at runtime.
- Solana support: Request Solana-specific chain/token data via
chainTypes=SVMand useSOLas the Solana chain key in API examples.
Product Surface Selection
Choose the smallest surface that matches the task:
- REST API: Backend services, scripts, and general AI agents. Simple HTTP calls, no runtime dependency.
- SDK (
@lifi/sdk): Frontend or full execution flows. Handles wallet connectors, signing, route execution, and update hooks. - MCP Server: MCP-compatible AI hosts such as Claude, Cursor, or Windsurf. Use when typed tool discovery is available.
- CLI: Token-efficient agent workflows where compact human-readable output is preferable to raw JSON.
- Widget: Ready-made UI when the user wants embedded swap/bridge UX rather than custom integration.
Do not force every integration through the SDK. For quote lookup, route comparison, or status checks, the API is often simpler and easier to audit.
Base URLs and Authentication
Production API: https://li.quest/v1
Staging API: https://staging.li.quest/v1
Docs: https://docs.li.fi
OpenAPI: https://docs.li.fi/openapi.yaml
LLM overview: https://docs.li.fi/llms.txt
LI.FI APIs can be used without an API key. Use an API key for higher rate limits or authenticated partner usage. Register an integration in the LI.FI Partner Portal to get an API key: https://portal.li.fi/
For direct REST calls, pass the key in the x-lifi-api-key header:
curl 'https://li.quest/v1/chains?chainTypes=EVM,SVM' \
--header 'accept: application/json' \
--header 'x-lifi-api-key: YOUR_API_KEY_IF_AVAILABLE'
Test a key server-side before using it in production:
curl 'https://li.quest/v1/keys/test' \
--header 'x-lifi-api-key: YOUR_API_KEY'
Never expose x-lifi-api-key in browser code, public repositories, or direct Widget configuration. If using the SDK from a backend or trusted runtime, pass the key through createConfig({ apiKey: '...' }); if using the Widget in a frontend, do not pass an API key.
Integration Workflow
-
Clarify the transfer intent
- Source chain, destination chain, source token, destination token.
- Amount in the token's smallest unit.
- Sender address and, when different, recipient address.
- Whether the user wants a single best route or route comparison.
-
Discover support instead of hardcoding
- Use
/chainsto verify chains. - Use
/tokensor/tokento verify token addresses and decimals. - Use
/toolsto list current bridges and exchanges. - Do not assume every token, bridge, or chain pair is available.
- Use
-
Choose quote vs routes
- Use
GET /quotefor a simple transfer where the best executable route is enough. - Use
POST /advanced/routeswhen comparing alternatives or when the user asks for route choice, cost, speed, tool allowlists, or multiple steps. - Use
POST /advanced/stepTransactionto populate transaction data for individual route steps when executing advanced routes.
- Use
-
Show the user what they will sign
- Summarize from-chain, to-chain, from-token, to-token, amount, estimated output, tool/bridge, fees, recipient, and slippage.
- Never ask a user to sign opaque transaction data without a human-readable summary.
-
Execute through the appropriate wallet path
- If using
GET /quote, the response already includestransactionRequest; after allowance/permit handling, submit that transaction with the source-chain wallet. - If using
POST /advanced/routes, first choose a route, then populate each step withPOST /advanced/stepTransactionbefore execution. - EVM transaction requests usually include fields such as
to,data,value, and gas fields. - Solana-originating transfers return Solana transaction data as base64 in
transactionRequest.data; deserialize, sign, and send through the user's Solana wallet or SDK path. - Prefer SDK
executeRoutefor production multi-step execution because it manages allowance and balance checks, chain switching, transaction data retrieval, transaction submission, and status tracking. - Never mutate
transactionRequest.data, calldata, recipient, refund, memo, or bridge-specific payloads after receiving them from LI.FI.
- If using
-
Track status after source-chain submission
- Poll
/statusevery 10-30 seconds until terminal status. - Include
fromChain,toChain, andbridgefrom the quote when available to speed up lookup. - Treat source-chain confirmation as only the start of a cross-chain transfer, not proof of final delivery.
- Poll
Minimal Endpoint Set
Get a quote
Use for a single best route with transaction data included.
curl --request GET \
--url 'https://li.quest/v1/quote?fromChain=ARB&toChain=SOL&fromToken=0xaf88d065e77c8cC2239327C5EDb3A432268e5831&toToken=7vfCXTUXx5WJV5JADk17DUJ4ksgau7utNKj4b963voxs&fromAddress=YOUR_EVM_WALLET&toAddress=YOUR_SOL_WALLET&fromAmount=1000000000' \
--header 'accept: application/json'
Required parameters:
fromChain: source chain ID or key.toChain: destination chain ID or key.fromToken: source token symbol or address.toToken: destination token symbol or address.fromAmount: amount in smallest unit.fromAddress: sender wallet address.toAddress: recipient wallet address when different from sender or when bridging across ecosystems.
Get multiple routes
Use when the user asks to compare routes or when the application needs route selection.
curl --request POST \
--url 'https://li.quest/v1/advanced/routes' \
--header 'accept: application/json' \
--header 'content-type: application/json' \
--data '{
"fromChainId": "ARB",
"toChainId": "SOL",
"fromTokenAddress": "0xaf88d065e77c8cC2239327C5EDb3A432268e5831",
"toTokenAddress": "7vfCXTUXx5WJV5JADk17DUJ4ksgau7utNKj4b963voxs",
"fromAmount": "1000000000",
"fromAddress": "YOUR_EVM_WALLET",
"toAddress": "YOUR_SOL_WALLET"
}'
Check transfer status
curl --request GET \
--url 'https://li.quest/v1/status?txHash=SOURCE_TX_HASH&fromChain=ARB&toChain=SOL&bridge=BRIDGE_KEY' \
--header 'accept: application/json'
Status handling:
NOT_FOUND: Transaction may not be indexed or mined yet. Retry withfromChainand bridge if known.PENDING: Continue polling.DONE+COMPLETED: Successful final delivery.DONE+PARTIAL: Successful but output token may differ while preserving value semantics.DONE+REFUNDED: Transfer failed but funds were refunded.FAILED: Stop polling and surface the error/substatus.
List supported chains
curl --request GET \
--url 'https://li.quest/v1/chains?chainTypes=EVM,SVM' \
--header 'accept: application/json'
Solana-only:
curl --request GET \
--url 'https://li.quest/v1/chains?chainTypes=SVM' \
--header 'accept: application/json'
List Solana tokens
curl --request GET \
--url 'https://li.quest/v1/tokens?chains=SOL&chainTypes=SVM' \
--header 'accept: application/json'
List bridges and exchanges
curl --request GET \
--url 'https://li.quest/v1/tools' \
--header 'accept: application/json'
Solana-Specific Guidance
Use this section whenever either side of the transfer is Solana.
- Use
chainTypes=SVMwhen querying Solana chain support. - Use
SOLas the Solana chain key in API examples. - Always pass a Solana
toAddresswhen bridging from an EVM chain to Solana; the sender's EVM address is not a valid Solana recipient. - For SOL -> EVM transfers, expect
transactionRequest.datato contain base64-encoded Solana transaction data. This is not EVM calldata. - Use a Solana wallet adapter or the LI.FI SDK to deserialize, sign, and submit Solana transactions. Do not manually rebuild the transaction unless the docs for the exact flow require it.
- Validate token decimals from
/tokensor/token; do not assume EVM and Solana versions of a token share decimals or addresses. - If status is slow, pass the source transaction hash plus
fromChain,toChain, and bridge/tool key from the quote.
Execution Patterns
Direct API execution
Use direct API execution when you control the wallet/provider layer yourself. /quote returns executable transaction data immediately; /advanced/routes returns route choices and requires POST /advanced/stepTransaction for each selected step.
For EVM-originating quotes, a typical flow is:
- Check token allowance for
quote.estimate.approvalAddresswhen the source token is not native. - Submit an approval or use Permit/Permit2 if required by the selected flow.
- Send
quote.transactionRequestthrough the EVM wallet. - Poll
/statuswith source transaction hash,quote.tool, source chain, and destination chain.
For Solana-originating quotes, do not treat transactionRequest.data as calldata. It is base64-encoded Solana transaction data; use the SDK or a Solana wallet-adapter/web3.js path to deserialize, sign, and submit it.
SDK execution
Prefer the SDK when building a frontend or when the agent needs full route execution rather than only quote/status lookup. The SDK exports createConfig, getQuote, getRoutes, executeRoute, EVM, Solana, and KeypairWalletAdapter from @lifi/sdk as of @lifi/sdk@3.16.3.
npm install @lifi/sdk
Quote-only lookup:
import { createConfig, getQuote } from '@lifi/sdk'
createConfig({
integrator: 'YourAppName',
// Backend/trusted runtime only. Do not expose this in browsers.
apiKey: process.env.LIFI_API_KEY,
})
const quote = await getQuote({
fromChain: 'ARB',
toChain: 'SOL',
fromToken: '0xaf88d065e77c8cC2239327C5EDb3A432268e5831',
toToken: '7vfCXTUXx5WJV5JADk17DUJ4ksgau7utNKj4b963voxs',
fromAmount: '1000000000',
fromAddress: 'YOUR_EVM_WALLET',
toAddress: 'YOUR_SOL_WALLET',
})
console.log(quote.estimate.toAmount, quote.tool, quote.transactionRequest)
Route execution with SDK:
import { createConfig, EVM, executeRoute, getRoutes } from '@lifi/sdk'
import type { Chain } from 'viem'
import { createWalletClient, http } from 'viem'
import { privateKeyToAccount } from 'viem/accounts'
import { arbitrum, mainnet, optimism } from 'viem/chains'
const privateKey = process.env.PRIVATE_KEY as `0x${string}` // backend/testing only
const account = privateKeyToAccount(privateKey)
const chains = [arbitrum, mainnet, optimism]
const client = createWalletClient({
account,
chain: arbitrum,
transport: http(),
})
createConfig({
integrator: 'YourAppName',
providers: [
EVM({
getWalletClient: async () => client,
switchChain: async (chainId) =>
createWalletClient({
account,
chain: chains.find((chain) => chain.id === chainId) as Chain,
transport: http(),
}),
}),
],
})
const result = await getRoutes({
fromChainId: 42161,
toChainId: 10,
fromTokenAddress: '0xaf88d065e77c8cC2239327C5EDb3A432268e5831',
toTokenAddress: '0xDA10009cBd5D07dd0CeCc66161FC93D7c9000da1',
fromAmount: '10000000',
fromAddress: 'YOUR_EVM_WALLET',
})
const executedRoute = await executeRoute(result.routes[0], {
updateRouteHook(route) {
console.log(route)
},
async acceptExchangeRateUpdateHook() {
// In a UI, ask the user before accepting.
return false
},
})
console.log(executedRoute)
Solana provider setup:
import { createConfig, KeypairWalletAdapter, Solana } from '@lifi/sdk'
const walletAdapter = new KeypairWalletAdapter('PRIVATE_KEY') // backend/testing only
createConfig({
integrator: 'YourAppName',
providers: [
Solana({
getWalletAdapter: async () => walletAdapter,
}),
],
})
For user-facing Solana apps, use @solana/wallet-adapter-react and set Solana({ getWalletAdapter }) from the connected wallet adapter at runtime. Do not embed private keys in frontend code.
Error Handling
Handle both HTTP status codes and LI.FI API error codes.
Common API error codes:
1002 NoQuoteError: No quote was found. Try another token, amount, destination, or route constraint.1005 RateLimitError: Back off and retry; use API key for higher limits if available.1007 SlippageError: Ask the user before increasing slippage.1008 ThirdPartyError: A bridge/exchange/tool failed. Retry later or request alternative routes.1009 TimeoutError: Retry with backoff or use route comparison to find another tool.1011 ValidationError: Check chain IDs, token addresses, amount units, and address format.1012 RpcFailure: Retry later or configure a more reliable RPC through the SDK when applicable.
Common tool error codes:
NO_POSSIBLE_ROUTE: No supported path for the requested action.INSUFFICIENT_LIQUIDITY: Reduce amount or choose another token/path.TOOL_TIMEOUT: Retry or exclude the timing-out tool if the API request supports tool constraints.AMOUNT_TOO_LOW/AMOUNT_TOO_HIGH: Adjust amount.FEES_HIGHER_THAN_AMOUNT: Amount is uneconomic; increase transfer size or choose another route.DIFFERENT_RECIPIENT_NOT_SUPPORTED: Use the same sender/recipient where required or choose another tool.CANNOT_GUARANTEE_MIN_AMOUNT: Explain the risk and ask the user before proceeding.
Guidelines
- DO query live chains, tokens, and tools before giving integration advice.
- DO use
/quotefor simple execution and/advanced/routesfor comparison. - DO show estimated output, fees, bridge/exchange tool, recipient, and slippage before signing.
- DO poll
/statusfor cross-chain transfers until a terminal state. - DO preserve LI.FI transaction data exactly as returned.
- DO register in the LI.FI Partner Portal for API keys when higher rate limits are needed, and keep keys server-side.
- DO NOT hardcode stale bridge lists, token lists, or chain support.
- DO NOT treat source-chain confirmation as final cross-chain completion.
- DO NOT increase slippage, change recipients, or alter transaction data without explicit user consent.
- DO NOT store or request private keys. Use wallet signing flows.
- DO NOT present the Widget as the only integration path when the API or SDK fits better.
Common Errors
Error: No route found
Cause: Unsupported chain/token pair, insufficient liquidity, amount too small or too large, or restrictive route options.
Solution: Verify /chains, /tokens, and /tools; try canonical token addresses; adjust amount; request /advanced/routes for alternatives.
Error: Solana transaction data treated like EVM calldata
Cause: A SOL-originating quote returns base64 Solana transaction data in transactionRequest.data, not an EVM to/data/value call.
Solution: Use a Solana wallet/SDK path to deserialize, sign, and submit the transaction. Do not send it through an EVM provider.
Error: Status endpoint returns NOT_FOUND
Cause: Transaction not indexed yet, wrong source transaction hash, missing fromChain, or wrong bridge/tool filter.
Solution: Retry with backoff and pass fromChain, toChain, and bridge from the quote. Confirm the hash is the source-chain transaction hash.
Error: Wrong amount due to decimals
Cause: The amount was not converted into the source token's smallest unit, or token decimals were assumed incorrectly across ecosystems.
Solution: Fetch token metadata from /tokens or /token and convert amounts before requesting a quote.
Error: Different recipient unsupported
Cause: Some tools do not support sending to a recipient different from the sender.
Solution: Request another route/tool or use the same recipient when acceptable to the user.
References
- Agent integration overview: https://docs.li.fi/agents/overview
- Agent concepts and object definitions: https://docs.li.fi/agents/concepts
- LLM-readable docs: https://docs.li.fi/llms.txt
- OpenAPI specification: https://docs.li.fi/openapi.yaml
- Quote API: https://docs.li.fi/api-reference/get-a-quote-for-a-token-transfer
- Routes API: https://docs.li.fi/api-reference/advanced/get-a-set-of-routes-for-a-request-that-describes-a-transfer-of-tokens
- Status API: https://docs.li.fi/api-reference/check-the-status-of-a-cross-chain-transfer
- Solana transaction execution: https://docs.li.fi/introduction/user-flows-and-examples/solana-tx-execution
- Error codes: https://docs.li.fi/api-reference/error-codes
- API introduction and authentication: https://docs.li.fi/api-reference/introduction
- Rate limits and API authentication: https://docs.li.fi/api-reference/rate-limits
- Partner Portal for API keys: https://portal.li.fi/
- SDK provider configuration: https://docs.li.fi/sdk/configure-sdk-providers
- SDK route requests: https://docs.li.fi/sdk/request-routes
- SDK route execution: https://docs.li.fi/sdk/execute-routes
More from sendaifun/skills
helius
Build Solana applications with Helius infrastructure. Covers transaction sending (Sender), asset/NFT queries (DAS API), real-time streaming (WebSockets, Laserstream), event pipelines (webhooks), priority fees, wallet analysis, and agent onboarding.
217solana-kit
Complete guide for @solana/kit - the modern, tree-shakeable, zero-dependency JavaScript SDK from Anza. Covers RPC connections, signers, transaction building with pipe, signing, sending, and account fetching with full TypeScript support.
203pumpfun
Complete PumpFun Protocol guide for building token launches, bonding curves, and AMM integrations on Solana. Covers Pump Program (token creation, buy/sell on bonding curves), PumpSwap AMM (liquidity pools, swaps), fee structures, creator fees, and SDK integration.
166meteora
Complete Meteora DeFi SDK suite for building liquidity pools, AMMs, bonding curves, vaults, token launches, and zap operations on Solana. Use when integrating DLMM, DAMM v2, DAMM v1, Dynamic Bonding Curves, Alpha Vaults, Zap, or Stake-for-Fee functionality.
165vulnhunter
Security vulnerability detection and variant analysis skill. Use when hunting for dangerous APIs, footgun patterns, error-prone configurations, and vulnerability variants across codebases. Combines sharp edges detection with variant hunting methodology.
147raydium
Complete Raydium Protocol SDK - the single source of truth for integrating Raydium on Solana. Covers SDK, Trade API, CLMM, CPMM, AMM pools, LaunchLab token launches, farming, CPI integration, and all Raydium tools.
145