solana_program_pack/lib.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63
//! The [`Pack`] serialization trait
//! This is a specific serialization API that is used by many older programs in
//! the [Solana Program Library][spl] to manage account state. It is not generally
//! recommended for new code since it does not define a language-independent
//! serialization format.
//!
//! [spl]: https://github.com/solana-labs/solana-program-library
use solana_program_error::ProgramError;
/// Check if a program account state is initialized
pub trait IsInitialized {
/// Is initialized
fn is_initialized(&self) -> bool;
}
/// Implementors must have a known size
pub trait Sealed: Sized {}
/// Safely and efficiently (de)serialize account state
pub trait Pack: Sealed {
/// The length, in bytes, of the packed representation
const LEN: usize;
#[doc(hidden)]
fn pack_into_slice(&self, dst: &mut [u8]);
#[doc(hidden)]
fn unpack_from_slice(src: &[u8]) -> Result<Self, ProgramError>;
/// Get the packed length
fn get_packed_len() -> usize {
Self::LEN
}
/// Unpack from slice and check if initialized
fn unpack(input: &[u8]) -> Result<Self, ProgramError>
where
Self: IsInitialized,
{
let value = Self::unpack_unchecked(input)?;
if value.is_initialized() {
Ok(value)
} else {
Err(ProgramError::UninitializedAccount)
}
}
/// Unpack from slice without checking if initialized
fn unpack_unchecked(input: &[u8]) -> Result<Self, ProgramError> {
if input.len() != Self::LEN {
return Err(ProgramError::InvalidAccountData);
}
Self::unpack_from_slice(input)
}
/// Pack into slice
fn pack(src: Self, dst: &mut [u8]) -> Result<(), ProgramError> {
if dst.len() != Self::LEN {
return Err(ProgramError::InvalidAccountData);
}
src.pack_into_slice(dst);
Ok(())
}
}