use {
crate::state::{Data, DurableNonce, State},
solana_hash::Hash,
solana_pubkey::Pubkey,
std::collections::HashSet,
};
#[cfg_attr(
feature = "serde",
derive(serde_derive::Deserialize, serde_derive::Serialize)
)]
#[derive(Debug, PartialEq, Eq, Clone)]
pub enum Versions {
Legacy(Box<State>),
Current(Box<State>),
}
#[derive(Debug, Eq, PartialEq)]
pub enum AuthorizeNonceError {
MissingRequiredSignature(Pubkey),
Uninitialized,
}
impl Versions {
pub fn new(state: State) -> Self {
Self::Current(Box::new(state))
}
pub fn state(&self) -> &State {
match self {
Self::Legacy(state) => state,
Self::Current(state) => state,
}
}
pub fn verify_recent_blockhash(
&self,
recent_blockhash: &Hash, ) -> Option<&Data> {
match self {
Self::Legacy(_) => None,
Self::Current(state) => match **state {
State::Uninitialized => None,
State::Initialized(ref data) => {
(recent_blockhash == &data.blockhash()).then_some(data)
}
},
}
}
pub fn upgrade(self) -> Option<Self> {
match self {
Self::Legacy(mut state) => {
match *state {
State::Uninitialized => None,
State::Initialized(ref mut data) => {
data.durable_nonce = DurableNonce::from_blockhash(&data.blockhash());
Some(Self::Current(state))
}
}
}
Self::Current(_) => None,
}
}
pub fn authorize(
self,
signers: &HashSet<Pubkey>,
authority: Pubkey,
) -> Result<Self, AuthorizeNonceError> {
let data = match self.state() {
State::Uninitialized => return Err(AuthorizeNonceError::Uninitialized),
State::Initialized(data) => data,
};
if !signers.contains(&data.authority) {
return Err(AuthorizeNonceError::MissingRequiredSignature(
data.authority,
));
}
let data = Data::new(
authority,
data.durable_nonce,
data.get_lamports_per_signature(),
);
let state = Box::new(State::Initialized(data));
Ok(match self {
Self::Legacy(_) => Self::Legacy,
Self::Current(_) => Self::Current,
}(state))
}
}
impl From<Versions> for State {
fn from(versions: Versions) -> Self {
match versions {
Versions::Legacy(state) => *state,
Versions::Current(state) => *state,
}
}
}
#[cfg(test)]
mod tests {
use {
super::*, solana_fee_calculator::FeeCalculator, solana_pubkey::Pubkey,
std::iter::repeat_with,
};
#[test]
fn test_verify_recent_blockhash() {
let blockhash = Hash::from([171; 32]);
let versions = Versions::Legacy(Box::new(State::Uninitialized));
assert_eq!(versions.verify_recent_blockhash(&blockhash), None);
assert_eq!(versions.verify_recent_blockhash(&Hash::default()), None);
let versions = Versions::Current(Box::new(State::Uninitialized));
assert_eq!(versions.verify_recent_blockhash(&blockhash), None);
assert_eq!(versions.verify_recent_blockhash(&Hash::default()), None);
let durable_nonce = DurableNonce::from_blockhash(&blockhash);
let data = Data {
authority: Pubkey::new_unique(),
durable_nonce,
fee_calculator: FeeCalculator {
lamports_per_signature: 2718,
},
};
let versions = Versions::Legacy(Box::new(State::Initialized(data.clone())));
assert_eq!(versions.verify_recent_blockhash(&Hash::default()), None);
assert_eq!(versions.verify_recent_blockhash(&blockhash), None);
assert_eq!(versions.verify_recent_blockhash(&data.blockhash()), None);
assert_eq!(
versions.verify_recent_blockhash(durable_nonce.as_hash()),
None
);
let durable_nonce = DurableNonce::from_blockhash(durable_nonce.as_hash());
assert_ne!(data.durable_nonce, durable_nonce);
let data = Data {
durable_nonce,
..data
};
let versions = Versions::Current(Box::new(State::Initialized(data.clone())));
assert_eq!(versions.verify_recent_blockhash(&blockhash), None);
assert_eq!(versions.verify_recent_blockhash(&Hash::default()), None);
assert_eq!(
versions.verify_recent_blockhash(&data.blockhash()),
Some(&data)
);
assert_eq!(
versions.verify_recent_blockhash(durable_nonce.as_hash()),
Some(&data)
);
}
#[test]
fn test_nonce_versions_upgrade() {
let versions = Versions::Legacy(Box::new(State::Uninitialized));
assert_eq!(versions.upgrade(), None);
let blockhash = Hash::from([171; 32]);
let durable_nonce = DurableNonce::from_blockhash(&blockhash);
let data = Data {
authority: Pubkey::new_unique(),
durable_nonce,
fee_calculator: FeeCalculator {
lamports_per_signature: 2718,
},
};
let versions = Versions::Legacy(Box::new(State::Initialized(data.clone())));
let durable_nonce = DurableNonce::from_blockhash(durable_nonce.as_hash());
assert_ne!(data.durable_nonce, durable_nonce);
let data = Data {
durable_nonce,
..data
};
let versions = versions.upgrade().unwrap();
assert_eq!(
versions,
Versions::Current(Box::new(State::Initialized(data)))
);
assert_eq!(versions.upgrade(), None);
}
#[test]
fn test_nonce_versions_authorize() {
let mut signers = repeat_with(Pubkey::new_unique).take(16).collect();
let versions = Versions::Legacy(Box::new(State::Uninitialized));
assert_eq!(
versions.authorize(&signers, Pubkey::new_unique()),
Err(AuthorizeNonceError::Uninitialized)
);
let versions = Versions::Current(Box::new(State::Uninitialized));
assert_eq!(
versions.authorize(&signers, Pubkey::new_unique()),
Err(AuthorizeNonceError::Uninitialized)
);
let blockhash = Hash::from([171; 32]);
let durable_nonce = DurableNonce::from_blockhash(&blockhash);
let data = Data {
authority: Pubkey::new_unique(),
durable_nonce,
fee_calculator: FeeCalculator {
lamports_per_signature: 2718,
},
};
let account_authority = data.authority;
let versions = Versions::Legacy(Box::new(State::Initialized(data.clone())));
let authority = Pubkey::new_unique();
assert_ne!(authority, account_authority);
let data = Data { authority, ..data };
assert_eq!(
versions.clone().authorize(&signers, authority),
Err(AuthorizeNonceError::MissingRequiredSignature(
account_authority
)),
);
assert!(signers.insert(account_authority));
assert_eq!(
versions.authorize(&signers, authority),
Ok(Versions::Legacy(Box::new(State::Initialized(data.clone()))))
);
let account_authority = data.authority;
let versions = Versions::Current(Box::new(State::Initialized(data.clone())));
let authority = Pubkey::new_unique();
assert_ne!(authority, account_authority);
let data = Data { authority, ..data };
assert_eq!(
versions.clone().authorize(&signers, authority),
Err(AuthorizeNonceError::MissingRequiredSignature(
account_authority
)),
);
assert!(signers.insert(account_authority));
assert_eq!(
versions.authorize(&signers, authority),
Ok(Versions::Current(Box::new(State::Initialized(data))))
);
}
}