Перейти к основному содержанию

Documentation Index

Fetch the complete documentation index at: https://docs.raydium.io/llms.txt

Use this file to discover all available pages before exploring further.

Эта страница переведена с помощью ИИ. За эталон принимается английская версия.Открыть английскую версию →
Что делает этот скрипт. Создает новый пул CPMM для двух выбранных вами токенов, выбирает комиссию 0,25%, предоставляет начальную ликвидность по цене, определяемой начальными объемами, и выводит ID нового пула и подпись транзакции.

Подготовка

Убедитесь, что вы прочитали Prerequisites быстрого старта и имеете установленные RPC_URL, KEYPAIR и зависимости. Также необходимо пополнить кошелек начальными объемами обоих токенов плюс достаточно SOL для однократной комиссии создания пула (примерно 0,15 SOL на mainnet, текущее значение см. в reference/program-addresses).

Скрипт

Сохраните как create-cpmm.mjs:
// create-cpmm.mjs
import { Connection, Keypair, PublicKey } from "@solana/web3.js";
import {
  Raydium,
  TxVersion,
  CREATE_CPMM_POOL_PROGRAM,
  CREATE_CPMM_POOL_FEE_ACC,
} from "@raydium-io/raydium-sdk-v2";
import BN from "bn.js";
import fs from "node:fs";

// ── Config from env ──────────────────────────────────────────────
const RPC_URL  = process.env.RPC_URL  ?? "https://api.mainnet-beta.solana.com";
const KEYPAIR  = process.env.KEYPAIR  ?? `${process.env.HOME}/.config/solana/id.json`;
const MINT_A   = process.env.MINT_A;   // required, base58
const MINT_B   = process.env.MINT_B;   // required, base58
const AMOUNT_A = process.env.AMOUNT_A; // required, integer raw units
const AMOUNT_B = process.env.AMOUNT_B; // required, integer raw units

if (!MINT_A || !MINT_B || !AMOUNT_A || !AMOUNT_B) {
  console.error("Set MINT_A, MINT_B, AMOUNT_A, AMOUNT_B env vars.");
  process.exit(1);
}

// ── Setup ────────────────────────────────────────────────────────
const connection = new Connection(RPC_URL, "confirmed");
const owner = Keypair.fromSecretKey(
  new Uint8Array(JSON.parse(fs.readFileSync(KEYPAIR, "utf8"))),
);
const raydium = await Raydium.load({
  owner,
  connection,
  cluster: "mainnet",
  disableFeatureCheck: true,
  blockhashCommitment: "finalized",
});

// ── Pick the 0.25% fee tier ─────────────────────────────────────
const feeConfigs = await raydium.api.getCpmmConfigs();
const feeConfig  = feeConfigs.find((c) => c.index === 0);
if (!feeConfig) throw new Error("0.25% fee tier not found in CPMM configs.");

// ── Resolve mint metadata (Token-2022-aware) ────────────────────
const mintA = await raydium.token.getTokenInfo(new PublicKey(MINT_A));
const mintB = await raydium.token.getTokenInfo(new PublicKey(MINT_B));

// ── Build and execute ───────────────────────────────────────────
const { execute, extInfo } = await raydium.cpmm.createPool({
  programId:       CREATE_CPMM_POOL_PROGRAM,
  poolFeeAccount:  CREATE_CPMM_POOL_FEE_ACC,
  mintA,
  mintB,
  mintAAmount:     new BN(AMOUNT_A),
  mintBAmount:     new BN(AMOUNT_B),
  startTime:       new BN(0), // open immediately
  feeConfig,
  associatedOnly:  false,
  ownerInfo:       { useSOLBalance: true },
  txVersion:       TxVersion.V0,
  computeBudgetConfig: {
    units: 600_000,
    microLamports: 100_000,
  },
});

const { txId } = await execute({ sendAndConfirm: true });

console.log(`Pool ID:  ${extInfo.address.poolId.toBase58()}`);
console.log(`LP mint:  ${extInfo.address.lpMint.toBase58()}`);
console.log(`Vault A:  ${extInfo.address.vaultA.toBase58()}`);
console.log(`Vault B:  ${extInfo.address.vaultB.toBase58()}`);
console.log(`Tx:       https://solscan.io/tx/${txId}`);

Запуск

Пример: создание пула SOL/USDC с начальным объемом 1 SOL и 160 USDC:
export MINT_A="So11111111111111111111111111111111111111112"   # wSOL
export MINT_B="EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v"   # USDC
export AMOUNT_A="1000000000"   # 1 SOL (9 decimals)
export AMOUNT_B="160000000"    # 160 USDC (6 decimals)

node create-cpmm.mjs
Ожидаемый результат:
Pool ID:  HgC5...3kXb
LP mint:  4ZAS...9rkV
Vault A:  J9Mu...mP2k
Vault B:  AaJq...8wxx
Tx:       https://solscan.io/tx/5dQ...

Что произошло

  1. getCpmmConfigs получила актуальный список уровней комиссии из api-v3.raydium.io и выбрала индекс 0 (уровень 0,25% — полный список см. в reference/fee-comparison).
  2. getTokenInfo разрешила метаданные каждого токена, включая информацию о владеющей им программе токенов. CPMM поддерживает как SPL Token, так и Token-2022; SDK выполняет маршрутизацию автоматически.
  3. createPool построила одну транзакцию, которая:
    • сортирует токены в каноническом порядке,
    • определяет PDA пула, хранилища, LP токен и органы управления,
    • платит однократную create_pool_fee в CREATE_CPMM_POOL_FEE_ACC,
    • создает ATA вызывающей стороны при необходимости,
    • предоставляет хранилища начальными объемами AMOUNT_A и AMOUNT_B.
  4. Начальная цена устанавливается по соотношению начальных объемов: цена = AMOUNT_B / AMOUNT_A после коррекции десятичных знаков. Выбирайте это внимательно — боты будут арбитражить любую неправильную цену в течение нескольких секунд после открытия пула.
  5. startTime: new BN(0) открывает торговлю немедленно. Для подготовки ликвидности перед открытием для публики установите будущую временную метку Unix.

Типичные ошибки

  • pool already exists — пул для этой пары токенов с этим уровнем комиссии уже существует. Проверьте его перед созданием.
  • insufficient funds — в кошельке недостаточно MINT_A, MINT_B или SOL (для комиссии создания пула + арендная плата).
  • Token-2022 extension not supported — один из токенов использует расширение, которое CPMM не поддерживает. См. reference/token-2022-support.

После развертывания

Вы можете сразу же выполнять обмены в новом пуле — скрипт Swap from CLI принимает ваш новый POOL_ID напрямую. Агрегаторы (Jupiter и прочие) индексируют новый пул в течение нескольких минут.

Дальше