本頁內容由 AI 自動翻譯,所有內容以英文版本為準。查看英文版 →
sdk-api/rust-cpi 涵蓋了呼叫各個 Raydium 程式的低階機制。本頁是更高階的補充:為什麼你會將 Raydium 組合到自己的程式中、哪種模式適合你的用例,以及端到端需要的完整膠合程式碼。CPI 適用的時機
當交易需要與只有你的程式才能進行的其他鏈上狀態變更原子性地發生時,自訂程式就很有用。常見的情況包括:- 託管帳戶/限價單程式 — 使用者將鑄幣存入你的託管帳戶,你的程式監視價格條件,當觸發時,你的程式原子性地透過 Raydium 進行交換並將結果記入使用者的帳戶。
- 匯總代理 — 單一指令可以將交換路由通過 Raydium 加上一個或多個其他 DEX,所有跳躍都在由你的程式擁有的單一滑點檢查下進行。
- 自動複合金庫 — 將 LP 或農場質押存入你的金庫,金庫按計劃收集獎勵、重新供應流動性、發行份額代幣。
- 策略金庫 — 槓桿 LP 頭寸,透過 CLMM 交換進行再平衡;清算人在一次交易中關閉頭寸並交換抵押品。
- 具有自訂解鎖的代幣發行平台 — 你的程式持有解鎖代幣並按計劃發放到 Raydium 池中。
組合模式
模式 1:薄代理
你的程式公開一個單一指令,驗證某些政策(例如白名單鑄幣對、經過驗證的使用者的費用折扣),然後轉發到 Raydium。模式 2:託管帳戶
你的程式擁有一個持有使用者輸入鑄幣的 PDA。在觸發時,PDA 簽署對 Raydium 進行交換的 CPI。CpiContext::new_with_signer 簽署。詳見 PDA 簽署人種子。
模式 3:複合多跳
你的程式在一個指令中發出多個 CPI,在所有 CPI 中強制執行單一滑點界限。Raydium 交換指令各自具有自己的minimum_amount_out,但你將它們設為 0(或非常鬆散的下限),並在最後一跳之後自己強制執行嚴格的最終最小值。
模式 4:金庫/策略
你的程式在 PDA 中持有 LP 代幣或農場質押。保管人(或使用者)呼叫compound(),該函式會:
- 從農場收集獎勵。
- 將獎勵交換為池代幣(透過 CPI 進入 CPMM 或 CLMM)。
- 將所得存入 LP(另一個 CPI)。
- 質押新的 LP(又一個 CPI)。
帳戶列表構造
呼叫程式的Accounts 結構鏡像 Raydium 程式的帳戶順序,但大多數 Raydium 端帳戶是 UncheckedAccount,因為 Raydium 會自己驗證它們。你只會在你擁有的帳戶上添加約束條件:
UncheckedAccount — 不是懶惰。接收器驗證自己的;在呼叫程式處進行雙重驗證只會燃燒 CU 並在 Raydium 發佈新的結構布局欄位時冒著不同步的風險。
CPI 呼叫本身
PDA 簽署人種子
CPI 只有在作為authority 傳遞的 PDA 與呼叫程式聲稱的推導相匹配時才會成功。兩者必須同意:
- 種子位元組序列(此處
[b"escrow", user.key().as_ref()])。 - bump。
- 呼叫程式 ID(你的程式,而不是 Raydium 的)。
authority 傳遞的簽名涵蓋交易且輸入 ATA 由該 authority 擁有。驗證發生在 anchor_spl::token::transfer 中:ATA 的 authority 欄位必須等於簽署人。
常見的錯誤:將 user 作為 authority 傳遞(並從由託管帳戶 PDA 擁有的 escrow_input_ata 轉賬)。SPL Token 程式會以 owner mismatch 拒絕。務必使 authority 欄位與 ATA 擁有者相符。
剩餘帳戶
幾個 Raydium 指令採用可變長度帳戶列表,附加在固定帳戶之後 — 剩餘帳戶。- CLMM
SwapV2:1–8 個TickArrayState帳戶,用於交換可能遍歷的刻度陣列,按交換方向。 - Farm v6
Deposit/Harvest/Withdraw:(reward_vault, user_reward_ata)對,每個活躍獎勵槽位一對。 - Token-2022 轉移鉤點鑄幣:轉移鉤點程式加上鉤點需要的任何帳戶。
複合呼叫的計算預算
CPI 本身的呼叫框架耗費約 1,500 CU;被呼叫者的自身 CU 用量堆疊在其上。每個 Raydium CPI 的粗略預算:| 呼叫 | CU (SPL Token) | CU (Token-2022) |
|---|---|---|
| CPMM swap_base_input | ~150,000 | ~200,000 |
| CLMM swap_v2 (單個刻度陣列) | ~180,000 | ~230,000 |
| CLMM swap_v2 (跨越 2 個刻度) | ~220,000 | ~270,000 |
| Farm v6 deposit | ~120,000 | ~150,000 |
| Farm v6 harvest (每個獎勵槽位) | +30,000 | +40,000 |
| AMM v4 swap_base_in | ~140,000 | n/a |
harvest → swap A → swap B → deposit LP → stake LP 的自動複合程式輕鬆達到 700k CU。
始終設定明確的 ComputeBudgetProgram::set_compute_unit_limit:
錯誤傳播
Raydium 的程式回傳具有穩定錯誤代碼的 Anchor 錯誤。你的呼叫程式會將它們視為Err(ProgramError::Custom(code))。預設冒泡:
sdk-api/anchor-idl)對每個版本是穩定的;新代碼附加在末尾,現有代碼的含義永遠不會改變。
完整的實作範例:限價單託管帳戶
流程:open_order— 使用者將amount_in個input_mint存入託管帳戶 PDA;記錄目標min_amount_out和到期時間。execute_order— 任何人(保管人)使用目前的池帳戶進行呼叫。程式檢查目前的報價 ≥min_amount_out,然後 CPI Raydium 交換並將輸出保留在託管帳戶中。claim— 使用者從託管帳戶提取輸出鑄幣。
測試
將 Raydium 程式拉入本地驗證器進行整合測試(來自Anchor.toml):
anchor test 在啟動時從主網取得它們。詳見 sdk-api/rust-cpi。
組合特有的陷阱
可重入性
Solana 沒有真正的可重入性 — CPI 無法在同一呼叫中呼叫回原始程式。但你仍然可以構建自己進入邏輯可重入性:CPI 讀取你的狀態,然後你的程式再次讀取它,假設 CPI 沒有改變它。對於 Raydium,CPI 不會碰你的狀態,所以這不如例如閃貸環境那樣令人擔憂。但如果你將 Raydium 與借貸協議組合,要注意。帳戶可變性漂移
如果你的程式將帳戶作為mut 傳遞,但 Raydium 期望它為唯讀(或反之),執行時會以 InvalidAccountData 拒絕呼叫。始終檢查 Raydium 的指令在 IDL 中的預期可變性;anchor_cp_swap::cpi::accounts::Swap 透過其欄位類型強制執行它。
Token-2022 程式欄位
輸入和輸出鑄幣可能在不同的代幣程式下 — 一個 SPL Token,一個 Token-2022。CPI 有分開的input_token_program 和 output_token_program 欄位,原因就在這裡。始終檢查每個鑄幣的 owner 欄位並將正確的程式路由到每個槽位。
版本化交易
包含 2+ 個 Raydium CPI 加上 ATA 建立的複合 tx 很少能放入傳統(v0-無-LUT)交易。使用 V0 和地址查找表;透過raydium.getRaydiumLutAddresses() 拉取 Raydium 的公共 LUT。
指標
sdk-api/rust-cpi— 低階 CPI 機制。integration-guides/priority-fee-tuning— 計算預算調整大小。products/cpmm/code-demos、products/clmm/code-demos、products/farm-staking/code-demos— 按產品的 CPI 程式碼片段。

