Files
libsam/pkg/shreder/program_raydiumlaunchlab.go

205 lines
6.7 KiB
Go
Raw Normal View History

package shreder
import (
"encoding/binary"
"fmt"
"github.com/gagliardetto/solana-go"
"github.com/shopspring/decimal"
)
var raydiumLaunchLabProgramID = solana.MustPublicKeyFromBase58("LanMV9sAd7wArD4vJFi2qDdfnVhFxYSUg6eADduJ3uj")
var (
raydiumLaunchLabInitializeV2PoolDiscriminator = []byte{67, 153, 175, 39, 218, 16, 38, 32}
raydiumLaunchLabInitializeWithToken2022PoolDiscriminator = []byte{37, 190, 126, 222, 44, 154, 171, 17}
raydiumLaunchLabSellExactInDiscriminator = []byte{0x95, 0x27, 0xde, 0x9b, 0xd3, 0x7c, 0x98, 0x1a}
raydiumLaunchLabSellExactOutDiscriminator = []byte{0x5f, 0xc8, 0x47, 0x22, 0x08, 0x09, 0x0b, 0xa6}
raydiumLaunchLabBuyExactInDiscriminator = []byte{0xfa, 0xea, 0x0d, 0x7b, 0xd5, 0x9c, 0x13, 0xec}
raydiumLaunchLabBuyExactOutDiscriminator = []byte{0x18, 0xd3, 0x74, 0x28, 0x69, 0x03, 0x99, 0x38}
)
func parseRaydiumLaunchLabInstruction(tx VersionedTransaction, instructionIndex int) (TxSignalBatch, error) {
if instructionIndex >= len(tx.Instructions) {
return nil, fmt.Errorf("instruction index out of bounds")
}
instruction := tx.Instructions[instructionIndex]
if len(instruction.Data) == 0 {
return nil, fmt.Errorf("data is empty")
}
if len(instruction.Data) < 8 {
return nil, nil
}
var (
err error
txSignal *TxSignal
)
if matchMethod(instruction.Data, raydiumLaunchLabBuyExactInDiscriminator) {
txSignal, err = parseRaydiumLaunchLabSwap(tx, instruction, true, true)
} else if matchMethod(instruction.Data, raydiumLaunchLabBuyExactOutDiscriminator) {
txSignal, err = parseRaydiumLaunchLabSwap(tx, instruction, true, false)
} else if matchMethod(instruction.Data, raydiumLaunchLabSellExactInDiscriminator) {
txSignal, err = parseRaydiumLaunchLabSwap(tx, instruction, false, true)
} else if matchMethod(instruction.Data, raydiumLaunchLabSellExactOutDiscriminator) {
txSignal, err = parseRaydiumLaunchLabSwap(tx, instruction, false, false)
} else if matchMethod(instruction.Data, raydiumLaunchLabInitializeV2PoolDiscriminator) {
txSignal, err = parseRaydiumLaunchLabCreate(tx, instruction, false)
} else if matchMethod(instruction.Data, raydiumLaunchLabInitializeWithToken2022PoolDiscriminator) {
txSignal, err = parseRaydiumLaunchLabCreate(tx, instruction, true)
}
if txSignal != nil {
return TxSignalBatch{txSignal}, err
}
return nil, err
}
func parseRaydiumLaunchLabCreate(tx VersionedTransaction, instruction Instructions, isToken2022 bool) (*TxSignal, error) {
if len(instruction.Accounts) < 10 {
return nil, fmt.Errorf("accounts too short")
}
2026-03-23 16:08:55 +08:00
base, err := tx.GetAccount(int(instruction.Accounts[6]))
if err != nil {
return nil, err
}
quote, err := tx.GetAccount(int(instruction.Accounts[7]))
if err != nil {
return nil, err
}
creator, err := tx.GetAccount(int(instruction.Accounts[1]))
if err != nil {
return nil, err
}
2026-03-23 16:08:55 +08:00
if !quote.Equals(wrappedSOL) {
// just ignore this
return nil, nil
}
return &TxSignal{
TxHash: tx.Signatures[0].String(),
Label: "raydiumlaunchlab",
Maker: creator.String(),
2026-03-23 16:08:55 +08:00
Token0Address: base.String(),
Token1Address: wsolMint,
Token0Amount: decimal.Zero,
Token1Amount: decimal.Zero,
Program: "RaydiumLaunchLab",
Event: "create",
IsToken2022: isToken2022,
IsMayhemMode: false,
Block: tx.Block,
Token0AmountUint64: 0,
Token1AmountUint64: 0,
}, nil
}
func parseRaydiumLaunchLabSwap(tx VersionedTransaction, instruction Instructions, isBuy bool, exactIn bool) (*TxSignal, error) {
if len(instruction.Accounts) < 10 {
return nil, fmt.Errorf("accounts too short")
}
if len(instruction.Data) < 24 {
return nil, fmt.Errorf("data too short for raydium launch lab swap args, len=%d", len(instruction.Data))
}
2026-03-23 16:08:55 +08:00
base, err := tx.GetAccount(int(instruction.Accounts[9]))
if err != nil {
return nil, err
}
quote, err := tx.GetAccount(int(instruction.Accounts[10]))
if err != nil {
return nil, err
}
user, err := tx.GetAccount(int(instruction.Accounts[0]))
if err != nil {
return nil, err
}
2026-03-23 16:08:55 +08:00
if !quote.Equals(wrappedSOL) {
// just ignore this
return nil, nil
}
amountA := binary.LittleEndian.Uint64(instruction.Data[8:16])
amountB := binary.LittleEndian.Uint64(instruction.Data[16:24])
if isBuy {
if exactIn {
return &TxSignal{
TxHash: tx.Signatures[0].String(),
Label: "raydiumlaunchlab",
Maker: user.String(),
2026-03-23 16:08:55 +08:00
Token0Address: base.String(),
Token1Address: wsolMint,
Token0Amount: formatTokenAmount(amountB),
Token1Amount: formatSolAmount(amountA),
Program: "RaydiumLaunchLab",
Event: "buy",
IsToken2022: false,
IsMayhemMode: false,
ExactSOL: true,
Block: tx.Block,
Token0AmountUint64: amountB,
Token1AmountUint64: amountA,
}, nil
} else {
return &TxSignal{
TxHash: tx.Signatures[0].String(),
Label: "raydiumlaunchlab",
Maker: user.String(),
2026-03-23 16:08:55 +08:00
Token0Address: base.String(),
Token1Address: wsolMint,
Token0Amount: formatTokenAmount(amountA),
Token1Amount: formatSolAmount(amountB),
Program: "RaydiumLaunchLab",
Event: "buy",
IsToken2022: false,
IsMayhemMode: false,
ExactSOL: false,
Block: tx.Block,
Token0AmountUint64: amountA,
Token1AmountUint64: amountB,
}, nil
}
} else {
if exactIn {
return &TxSignal{
TxHash: tx.Signatures[0].String(),
Label: "raydiumlaunchlab",
Maker: user.String(),
2026-03-23 16:08:55 +08:00
Token0Address: base.String(),
Token1Address: wsolMint,
Token0Amount: formatTokenAmount(amountA),
Token1Amount: formatSolAmount(amountB),
Program: "RaydiumLaunchLab",
Event: "sell",
IsToken2022: false,
IsMayhemMode: false,
ExactSOL: false,
Block: tx.Block,
Token0AmountUint64: amountA,
Token1AmountUint64: amountB,
}, nil
} else {
return &TxSignal{
TxHash: tx.Signatures[0].String(),
Label: "raydiumlaunchlab",
Maker: user.String(),
2026-03-23 16:08:55 +08:00
Token0Address: base.String(),
Token1Address: wsolMint,
Token0Amount: formatTokenAmount(amountB),
Token1Amount: formatSolAmount(amountA),
Program: "RaydiumLaunchLab",
Event: "sell",
IsToken2022: false,
IsMayhemMode: false,
ExactSOL: true,
Block: tx.Block,
Token0AmountUint64: amountB,
Token1AmountUint64: amountA,
}, nil
}
}
}