use {
super::{Node, Snapshot},
crate::{
errors::ClockworkError,
objects::{NodeAccount, SnapshotAccount, SnapshotStatus},
},
anchor_lang::{prelude::*, AnchorDeserialize},
anchor_spl::token::TokenAccount,
std::convert::TryFrom,
};
const SEED_REGISTRY: &[u8] = b"registry";
#[account]
#[derive(Debug)]
pub struct Registry {
pub is_locked: bool,
pub node_count: u64,
pub snapshot_count: u64,
}
impl Registry {
pub fn pubkey() -> Pubkey {
Pubkey::find_program_address(&[SEED_REGISTRY], &crate::ID).0
}
}
impl TryFrom<Vec<u8>> for Registry {
type Error = Error;
fn try_from(data: Vec<u8>) -> std::result::Result<Self, Self::Error> {
Registry::try_deserialize(&mut data.as_slice())
}
}
pub trait RegistryAccount {
fn init(&mut self) -> Result<()>;
fn new_node(
&mut self,
authority: &mut Signer,
node: &mut Account<Node>,
stake: &mut Account<TokenAccount>,
worker: &Signer,
) -> Result<()>;
fn new_snapshot(&mut self, snapshot: &mut Account<Snapshot>) -> Result<()>;
fn rotate_snapshot(
&mut self,
current_snapshot: Option<&mut Account<Snapshot>>,
next_snapshot: &mut Account<Snapshot>,
) -> Result<()>;
fn lock(&mut self) -> Result<()>;
fn unlock(&mut self) -> Result<()>;
}
impl RegistryAccount for Account<'_, Registry> {
fn init(&mut self) -> Result<()> {
self.is_locked = false;
self.node_count = 0;
self.snapshot_count = 0;
Ok(())
}
fn new_node(
&mut self,
authority: &mut Signer,
node: &mut Account<Node>,
stake: &mut Account<TokenAccount>,
worker: &Signer,
) -> Result<()> {
require!(!self.is_locked, ClockworkError::RegistryLocked);
node.init(authority, self.node_count, stake, worker)?;
self.node_count = self.node_count.checked_add(1).unwrap();
Ok(())
}
fn new_snapshot(&mut self, snapshot: &mut Account<Snapshot>) -> Result<()> {
require!(!self.is_locked, ClockworkError::RegistryLocked);
self.lock()?;
snapshot.init(self.snapshot_count)?;
Ok(())
}
fn rotate_snapshot(
&mut self,
current_snapshot: Option<&mut Account<Snapshot>>,
next_snapshot: &mut Account<Snapshot>,
) -> Result<()> {
require!(self.is_locked, ClockworkError::RegistryMustBeLocked);
require!(
next_snapshot.status == SnapshotStatus::InProgress,
ClockworkError::SnapshotNotInProgress
);
require!(
next_snapshot.node_count == self.node_count,
ClockworkError::SnapshotIncomplete
);
match current_snapshot {
Some(current_snapshot) => {
require!(
current_snapshot.status == SnapshotStatus::Current,
ClockworkError::SnapshotNotCurrent
);
current_snapshot.status = SnapshotStatus::Archived;
}
None => require!(self.snapshot_count == 0, ClockworkError::SnapshotNotCurrent),
}
next_snapshot.status = SnapshotStatus::Current;
self.snapshot_count = self.snapshot_count.checked_add(1).unwrap();
self.unlock()?;
Ok(())
}
fn lock(&mut self) -> Result<()> {
self.is_locked = true;
Ok(())
}
fn unlock(&mut self) -> Result<()> {
self.is_locked = false;
Ok(())
}
}