solana-router-raydium 1.1.3

Solana Raydium Router
Documentation
//! Unstake LP tokens from a Raydium farm instruction

use {
    solana_farm_sdk::{
        id::zero,
        instruction::raydium::RaydiumUnstake,
        program::{account, protocol::raydium},
    },
    solana_program::{
        account_info::AccountInfo,
        entrypoint::ProgramResult,
        instruction::{AccountMeta, Instruction},
        msg,
        program::invoke,
        program_error::ProgramError,
    },
};

pub fn unstake(accounts: &[AccountInfo], amount: u64) -> ProgramResult {
    msg!("Processing AmmInstruction::Unstake");
    msg!("amount {} ", amount);

    #[allow(clippy::deprecated_cfg_attr)]
    #[cfg_attr(rustfmt, rustfmt_skip)]
    if let [
        user_account,
        user_info_account,
        user_lp_token_account,
        user_first_reward_token_account,
        user_second_reward_token_account,
        farm_program_id,
        farm_lp_token_account,
        farm_first_reward_token_account,
        farm_second_reward_token_account,
        clock_id,
        spl_token_id,
        farm_id,
        farm_authority
        ] = accounts
    {
        if !raydium::check_stake_program_id(farm_program_id.key) {
            return Err(ProgramError::IncorrectProgramId);
        }
        if !account::check_token_account_owner(user_lp_token_account, user_account.key)?
            || !account::check_token_account_owner(user_first_reward_token_account, user_account.key)?
            || !account::check_token_account_owner_or_zero(user_second_reward_token_account, user_account.key)?
        {
            return Err(ProgramError::IllegalOwner);
        }

        let dual_rewards = *farm_second_reward_token_account.key != zero::id();
        let initial_token_a_user_balance = account::get_token_balance(user_first_reward_token_account)?;
        let initial_token_b_user_balance = if dual_rewards {
            account::get_token_balance(user_second_reward_token_account)?
        } else {
            0
        };
        let initial_lp_token_user_balance = account::get_token_balance(user_lp_token_account)?;

        let mut raydium_accounts = Vec::with_capacity(12);
        raydium_accounts.push(AccountMeta::new(*farm_id.key, false));
        raydium_accounts.push(AccountMeta::new_readonly(*farm_authority.key, false));
        raydium_accounts.push(AccountMeta::new(*user_info_account.key, false));
        raydium_accounts.push(AccountMeta::new_readonly(*user_account.key, true));
        raydium_accounts.push(AccountMeta::new(*user_lp_token_account.key, false));
        raydium_accounts.push(AccountMeta::new(*farm_lp_token_account.key, false));
        raydium_accounts.push(AccountMeta::new(*user_first_reward_token_account.key, false));
        raydium_accounts.push(AccountMeta::new(*farm_first_reward_token_account.key, false));
        raydium_accounts.push(AccountMeta::new_readonly(*clock_id.key, false));
        raydium_accounts.push(AccountMeta::new_readonly(*spl_token_id.key, false));
        if dual_rewards {
            raydium_accounts.push(AccountMeta::new(*user_second_reward_token_account.key, false));
            raydium_accounts.push(AccountMeta::new(*farm_second_reward_token_account.key, false));
        }

        let lp_amount = if amount > 0 {
            amount
        } else {
            raydium::get_stake_account_balance(user_info_account)?
        };

        let instruction = Instruction {
            program_id: *farm_program_id.key,
            accounts: raydium_accounts,
            data: RaydiumUnstake {
                instruction: 2,
                amount: lp_amount,
            }
            .to_vec()?,
        };
        invoke(&instruction, accounts)?;

        account::check_tokens_received(
            user_lp_token_account,
            initial_lp_token_user_balance,
            lp_amount,
        )?;
        let _ = account::get_balance_increase(
            user_first_reward_token_account,
            initial_token_a_user_balance,
        )?;
        if dual_rewards {
            let _ = account::get_balance_increase(
                user_second_reward_token_account,
                initial_token_b_user_balance,
            )?;
        }
    } else {
        return Err(ProgramError::NotEnoughAccountKeys);
    }

    msg!("AmmInstruction::Unstake complete");
    Ok(())
}