use anchor_lang::{prelude::*, InstructionData, solana_program::instruction::Instruction};
use clockwork_utils::thread::ThreadResponse;
use crate::state::*;
#[derive(Accounts)]
pub struct DeleteSnapshotProcessEntry<'info> {
#[account(address = Config::pubkey())]
pub config: Account<'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.lt(®istry.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.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.to_account_info().lamports();
**snapshot_entry.to_account_info().lamports.borrow_mut() = 0;
**thread.to_account_info().lamports.borrow_mut() = thread
.to_account_info()
.lamports()
.checked_add(snapshot_entry_lamports)
.unwrap();
if snapshot_entry.id.checked_add(1).unwrap().eq(&snapshot_frame.total_entries) {
let snapshot_frame_lamports = snapshot_frame.to_account_info().lamports();
**snapshot_frame.to_account_info().lamports.borrow_mut() = 0;
**thread.to_account_info().lamports.borrow_mut() = thread
.to_account_info()
.lamports()
.checked_add(snapshot_frame_lamports)
.unwrap();
if snapshot_frame.id.checked_add(1).unwrap().eq(&snapshot.total_frames) {
let snapshot_lamports = snapshot.to_account_info().lamports();
**snapshot.to_account_info().lamports.borrow_mut() = 0;
**thread.to_account_info().lamports.borrow_mut() = thread
.to_account_info()
.lamports()
.checked_add(snapshot_lamports)
.unwrap();
}
}
let dynamic_instruction = if snapshot_entry.id.checked_add(1).unwrap().lt(&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.checked_add(1).unwrap()),
snapshot_frame:snapshot_frame.key(),
thread: thread.key(),
}.to_account_metas(Some(true)),
data: crate::instruction::DeleteSnapshotProcessEntry{}.data()
}.into()
)
} else if snapshot_frame.id.checked_add(1).unwrap().lt(&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.checked_add(1).unwrap()),
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 } )
}