anvil_zksync_core/node/
storage_logs.rsuse std::collections::HashMap;
use crate::formatter::{self, PubdataBytesInfo};
use anvil_zksync_common::sh_println;
use anvil_zksync_types::ShowStorageLogs;
use zksync_multivm::interface::VmExecutionResultAndLogs;
use zksync_types::h256_to_u256;
use zksync_types::{
utils::storage_key_for_eth_balance,
writes::{
compression::compress_with_best_strategy, BYTES_PER_DERIVED_KEY,
BYTES_PER_ENUMERATION_INDEX,
},
StorageKey, BOOTLOADER_ADDRESS, SYSTEM_CONTEXT_ADDRESS,
};
use zksync_types::{StorageLogKind, StorageLogWithPreviousValue};
fn is_storage_key_free(key: &StorageKey) -> bool {
key.address() == &SYSTEM_CONTEXT_ADDRESS
|| *key == storage_key_for_eth_balance(&BOOTLOADER_ADDRESS)
}
fn compute_and_update_pubdata_cost(
cost_paid: &mut HashMap<StorageKey, u32>,
log_query: &StorageLogWithPreviousValue,
) -> PubdataBytesInfo {
let storage_key = log_query.log.key;
if is_storage_key_free(&storage_key) {
PubdataBytesInfo::FreeSlot
} else {
let compressed_value_size = compress_with_best_strategy(
h256_to_u256(log_query.previous_value),
h256_to_u256(log_query.log.value),
)
.len() as u32;
let final_pubdata_cost = if log_query.log.kind == StorageLogKind::InitialWrite {
(BYTES_PER_DERIVED_KEY as u32) + compressed_value_size
} else {
(BYTES_PER_ENUMERATION_INDEX as u32) + compressed_value_size
};
let result = match cost_paid.get(&storage_key).copied() {
Some(already_paid) => {
let to_pay = final_pubdata_cost.saturating_sub(already_paid);
if to_pay > 0 {
PubdataBytesInfo::AdditionalPayment(to_pay, final_pubdata_cost)
} else {
PubdataBytesInfo::PaidAlready
}
}
None => PubdataBytesInfo::Paid(final_pubdata_cost),
};
cost_paid.insert(storage_key, final_pubdata_cost);
result
}
}
pub fn print_storage_logs_details(
show_storage_logs: ShowStorageLogs,
result: &VmExecutionResultAndLogs,
) {
sh_println!("\n");
sh_println!(
"[Storage Logs] ({} entries)",
result.logs.storage_logs.len()
);
let mut cost_paid = HashMap::<StorageKey, u32>::default();
for (index, log_query) in result.logs.storage_logs.iter().enumerate() {
let pubdata_bytes_info = if matches!(
log_query.log.kind,
StorageLogKind::RepeatedWrite | StorageLogKind::InitialWrite
) {
Some(compute_and_update_pubdata_cost(&mut cost_paid, log_query))
} else {
None
};
let should_print = match show_storage_logs {
ShowStorageLogs::Write => matches!(
log_query.log.kind,
StorageLogKind::RepeatedWrite | StorageLogKind::InitialWrite
),
ShowStorageLogs::Paid => pubdata_bytes_info
.as_ref()
.map(|x| x.does_cost())
.unwrap_or_default(),
ShowStorageLogs::Read => log_query.log.kind == StorageLogKind::Read,
ShowStorageLogs::All => true,
_ => false,
};
if should_print {
let is_last = index == result.logs.storage_logs.len() - 1;
let mut formatter = formatter::Formatter::new();
formatter.print_storage_logs(log_query, pubdata_bytes_info, index + 1, is_last);
}
}
}