use anchor_lang::{prelude::*, solana_program::instruction::Instruction, InstructionData};
use sablier_utils::thread::ThreadResponse;
use crate::{constants::*, state::*};
#[derive(Accounts)]
pub struct DeleteSnapshotProcessEntry<'info> {
#[account(address = Config::pubkey())]
pub config: AccountLoader<'info, Config>,
#[account(
address = Registry::pubkey(),
constraint = !registry.locked
)]
pub registry: Account<'info, Registry>,
#[account(
mut,
seeds = [
SEED_SNAPSHOT,
snapshot.id.to_be_bytes().as_ref(),
],
bump,
constraint = snapshot.id < registry.current_epoch
)]
pub snapshot: Account<'info, Snapshot>,
#[account(
mut,
seeds = [
SEED_SNAPSHOT_ENTRY,
snapshot_entry.snapshot_frame.as_ref(),
snapshot_entry.id.to_be_bytes().as_ref(),
],
bump,
has_one = snapshot_frame
)]
pub snapshot_entry: Account<'info, SnapshotEntry>,
#[account(
mut,
seeds = [
SEED_SNAPSHOT_FRAME,
snapshot_frame.snapshot.as_ref(),
snapshot_frame.id.to_be_bytes().as_ref(),
],
bump,
has_one = snapshot,
)]
pub snapshot_frame: Account<'info, SnapshotFrame>,
#[account(
mut,
address = config.load()?.epoch_thread
)]
pub thread: Signer<'info>,
}
pub fn handler(ctx: Context<DeleteSnapshotProcessEntry>) -> Result<ThreadResponse> {
let config = &ctx.accounts.config;
let registry = &ctx.accounts.registry;
let snapshot = &mut ctx.accounts.snapshot;
let snapshot_entry = &mut ctx.accounts.snapshot_entry;
let snapshot_frame = &mut ctx.accounts.snapshot_frame;
let thread = &mut ctx.accounts.thread;
let snapshot_entry_lamports = snapshot_entry.get_lamports();
snapshot_entry.sub_lamports(snapshot_entry_lamports)?;
thread.add_lamports(snapshot_entry_lamports)?;
if (snapshot_entry.id + 1) == snapshot_frame.total_entries {
let snapshot_frame_lamports = snapshot_frame.get_lamports();
snapshot_frame.sub_lamports(snapshot_frame_lamports)?;
thread.add_lamports(snapshot_frame_lamports)?;
if (snapshot_frame.id + 1) == snapshot.total_frames {
let snapshot_lamports = snapshot.get_lamports();
snapshot.sub_lamports(snapshot_lamports)?;
thread.add_lamports(snapshot_frame_lamports)?;
}
}
let dynamic_instruction = if (snapshot_entry.id + 1) < snapshot_frame.total_entries {
Some(
Instruction {
program_id: crate::ID,
accounts: crate::accounts::DeleteSnapshotProcessEntry {
config: config.key(),
registry: registry.key(),
snapshot: snapshot.key(),
snapshot_entry: SnapshotEntry::pubkey(
snapshot_frame.key(),
snapshot_entry.id + 1,
),
snapshot_frame: snapshot_frame.key(),
thread: thread.key(),
}
.to_account_metas(Some(true)),
data: crate::instruction::DeleteSnapshotProcessEntry {}.data(),
}
.into(),
)
} else if (snapshot_frame.id + 1) < snapshot.total_frames {
Some(
Instruction {
program_id: crate::ID,
accounts: crate::accounts::DeleteSnapshotProcessFrame {
config: config.key(),
registry: registry.key(),
snapshot: snapshot.key(),
snapshot_frame: SnapshotFrame::pubkey(snapshot.key(), snapshot_frame.id + 1),
thread: thread.key(),
}
.to_account_metas(Some(true)),
data: crate::instruction::DeleteSnapshotProcessFrame {}.data(),
}
.into(),
)
} else {
None
};
Ok(ThreadResponse {
dynamic_instruction,
close_to: None,
trigger: None,
})
}