use alloc::vec::Vec;
use super::{Account, AccountId, Felt, PartialAccount};
use crate::crypto::SequentialCommit;
use crate::errors::AccountError;
use crate::transaction::memory::{
ACCT_CODE_COMMITMENT_OFFSET,
ACCT_DATA_MEM_SIZE,
ACCT_ID_AND_NONCE_OFFSET,
ACCT_ID_PREFIX_IDX,
ACCT_ID_SUFFIX_IDX,
ACCT_NONCE_IDX,
ACCT_STORAGE_COMMITMENT_OFFSET,
ACCT_VAULT_ROOT_OFFSET,
MemoryOffset,
};
use crate::utils::serde::{
ByteReader,
ByteWriter,
Deserializable,
DeserializationError,
Serializable,
};
use crate::{WORD_SIZE, Word, WordError};
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct AccountHeader {
id: AccountId,
nonce: Felt,
vault_root: Word,
storage_commitment: Word,
code_commitment: Word,
}
impl AccountHeader {
pub fn new(
id: AccountId,
nonce: Felt,
vault_root: Word,
storage_commitment: Word,
code_commitment: Word,
) -> Self {
Self {
id,
nonce,
vault_root,
storage_commitment,
code_commitment,
}
}
pub(crate) fn try_from_elements(elements: &[Felt]) -> Result<AccountHeader, AccountError> {
if elements.len() != ACCT_DATA_MEM_SIZE {
return Err(AccountError::HeaderDataIncorrectLength {
actual: elements.len(),
expected: ACCT_DATA_MEM_SIZE,
});
}
let id = AccountId::try_from_elements(
elements[ACCT_ID_AND_NONCE_OFFSET as usize + ACCT_ID_SUFFIX_IDX],
elements[ACCT_ID_AND_NONCE_OFFSET as usize + ACCT_ID_PREFIX_IDX],
)
.map_err(AccountError::FinalAccountHeaderIdParsingFailed)?;
let nonce = elements[ACCT_ID_AND_NONCE_OFFSET as usize + ACCT_NONCE_IDX];
let vault_root = parse_word(elements, ACCT_VAULT_ROOT_OFFSET)
.expect("we should have sliced off exactly 4 bytes");
let storage_commitment = parse_word(elements, ACCT_STORAGE_COMMITMENT_OFFSET)
.expect("we should have sliced off exactly 4 bytes");
let code_commitment = parse_word(elements, ACCT_CODE_COMMITMENT_OFFSET)
.expect("we should have sliced off exactly 4 bytes");
Ok(AccountHeader::new(id, nonce, vault_root, storage_commitment, code_commitment))
}
pub fn to_commitment(&self) -> Word {
<Self as SequentialCommit>::to_commitment(self)
}
pub fn id(&self) -> AccountId {
self.id
}
pub fn nonce(&self) -> Felt {
self.nonce
}
pub fn vault_root(&self) -> Word {
self.vault_root
}
pub fn storage_commitment(&self) -> Word {
self.storage_commitment
}
pub fn code_commitment(&self) -> Word {
self.code_commitment
}
pub fn to_elements(&self) -> Vec<Felt> {
<Self as SequentialCommit>::to_elements(self)
}
}
impl From<PartialAccount> for AccountHeader {
fn from(account: PartialAccount) -> Self {
(&account).into()
}
}
impl From<&PartialAccount> for AccountHeader {
fn from(account: &PartialAccount) -> Self {
Self {
id: account.id(),
nonce: account.nonce(),
vault_root: account.vault().root(),
storage_commitment: account.storage().commitment(),
code_commitment: account.code().commitment(),
}
}
}
impl From<Account> for AccountHeader {
fn from(account: Account) -> Self {
(&account).into()
}
}
impl From<&Account> for AccountHeader {
fn from(account: &Account) -> Self {
Self {
id: account.id(),
nonce: account.nonce(),
vault_root: account.vault().root(),
storage_commitment: account.storage().to_commitment(),
code_commitment: account.code().commitment(),
}
}
}
impl SequentialCommit for AccountHeader {
type Commitment = Word;
fn to_elements(&self) -> Vec<Felt> {
let mut id_nonce = Word::empty();
id_nonce[ACCT_NONCE_IDX] = self.nonce;
id_nonce[ACCT_ID_SUFFIX_IDX] = self.id.suffix();
id_nonce[ACCT_ID_PREFIX_IDX] = self.id.prefix().as_felt();
[
id_nonce.as_elements(),
self.vault_root.as_elements(),
self.storage_commitment.as_elements(),
self.code_commitment.as_elements(),
]
.concat()
}
}
impl Serializable for AccountHeader {
fn write_into<W: ByteWriter>(&self, target: &mut W) {
self.id.write_into(target);
self.nonce.write_into(target);
self.vault_root.write_into(target);
self.storage_commitment.write_into(target);
self.code_commitment.write_into(target);
}
}
impl Deserializable for AccountHeader {
fn read_from<R: ByteReader>(source: &mut R) -> Result<Self, DeserializationError> {
let id = AccountId::read_from(source)?;
let nonce = Felt::read_from(source)?;
let vault_root = Word::read_from(source)?;
let storage_commitment = Word::read_from(source)?;
let code_commitment = Word::read_from(source)?;
Ok(AccountHeader {
id,
nonce,
vault_root,
storage_commitment,
code_commitment,
})
}
}
fn parse_word(data: &[Felt], offset: MemoryOffset) -> Result<Word, WordError> {
Word::try_from(&data[offset as usize..offset as usize + WORD_SIZE])
}
#[cfg(test)]
mod tests {
use miden_core::Felt;
use super::AccountHeader;
use crate::Word;
use crate::account::StorageSlotContent;
use crate::account::tests::build_account;
use crate::asset::FungibleAsset;
use crate::utils::serde::{Deserializable, Serializable};
#[test]
fn test_serde_account_storage() {
let init_nonce = Felt::new(1);
let asset_0 = FungibleAsset::mock(99);
let word = Word::from([1, 2, 3, 4u32]);
let storage_slot = StorageSlotContent::Value(word);
let account = build_account(vec![asset_0], init_nonce, vec![storage_slot]);
let account_header: AccountHeader = account.into();
let header_bytes = account_header.to_bytes();
let deserialized_header = AccountHeader::read_from_bytes(&header_bytes).unwrap();
assert_eq!(deserialized_header, account_header);
}
}