anvil_zksync_core/node/
storage_logs.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
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
use 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 {
        // how many bytes it takes after compression.
        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
        };

        // Filter logs based on the selected storage log type
        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);
        }
    }
}