use std::mem;
use aligned_sized::aligned_sized;
use anchor_lang::prelude::*;
use light_concurrent_merkle_tree::{
    zero_copy::{ConcurrentMerkleTreeZeroCopy, ConcurrentMerkleTreeZeroCopyMut},
    ConcurrentMerkleTree,
};
use light_hasher::Poseidon;
use crate::{AccessMetadata, MerkleTreeMetadata, RolloverMetadata};
#[account(zero_copy)]
#[aligned_sized(anchor)]
#[derive(AnchorDeserialize, Debug, Default)]
pub struct StateMerkleTreeAccount {
    pub metadata: MerkleTreeMetadata,
}
impl StateMerkleTreeAccount {
    pub fn size(
        height: usize,
        changelog_size: usize,
        roots_size: usize,
        canopy_depth: usize,
    ) -> usize {
        8 + mem::size_of::<Self>()
            + ConcurrentMerkleTree::<Poseidon, 26>::size_in_account(
                height,
                changelog_size,
                roots_size,
                canopy_depth,
            )
    }
    pub fn init(
        &mut self,
        access_metadata: AccessMetadata,
        rollover_metadata: RolloverMetadata,
        associated_queue: Pubkey,
    ) {
        self.metadata
            .init(access_metadata, rollover_metadata, associated_queue)
    }
}
pub fn state_merkle_tree_from_bytes_zero_copy_init(
    data: &mut [u8],
    height: usize,
    canopy_depth: usize,
    changelog_capacity: usize,
    roots_capacity: usize,
) -> Result<ConcurrentMerkleTreeZeroCopyMut<Poseidon, 26>> {
    let data = &mut data[8 + mem::size_of::<StateMerkleTreeAccount>()..];
    let merkle_tree = ConcurrentMerkleTreeZeroCopyMut::from_bytes_zero_copy_init(
        data,
        height,
        canopy_depth,
        changelog_capacity,
        roots_capacity,
    )
    .map_err(ProgramError::from)?;
    Ok(merkle_tree)
}
pub fn state_merkle_tree_from_bytes_zero_copy(
    data: &[u8],
) -> Result<ConcurrentMerkleTreeZeroCopy<Poseidon, 26>> {
    let data = &data[8 + mem::size_of::<StateMerkleTreeAccount>()..];
    let merkle_tree =
        ConcurrentMerkleTreeZeroCopy::from_bytes_zero_copy(data).map_err(ProgramError::from)?;
    Ok(merkle_tree)
}
pub fn state_merkle_tree_from_bytes_zero_copy_mut(
    data: &mut [u8],
) -> Result<ConcurrentMerkleTreeZeroCopyMut<Poseidon, 26>> {
    let data = &mut data[8 + mem::size_of::<StateMerkleTreeAccount>()..];
    let merkle_tree = ConcurrentMerkleTreeZeroCopyMut::from_bytes_zero_copy_mut(data)
        .map_err(ProgramError::from)?;
    Ok(merkle_tree)
}