Skip to main content

ethrex_storage/
trie.rs

1use crate::api::tables::{
2    ACCOUNT_FLATKEYVALUE, ACCOUNT_TRIE_NODES, STORAGE_FLATKEYVALUE, STORAGE_TRIE_NODES,
3};
4use crate::api::{StorageBackend, StorageLockedView, StorageReadView};
5use crate::error::StoreError;
6use crate::layering::apply_prefix;
7use ethrex_common::H256;
8use ethrex_trie::{Nibbles, TrieDB, error::TrieError};
9use std::sync::Arc;
10
11/// TrieDB implementation that holds a pre-acquired read view for the entire
12/// trie traversal, avoiding per-node-lookup allocation and lock acquisition.
13pub struct BackendTrieDB {
14    /// Reference to the storage backend (used only for writes)
15    db: Arc<dyn StorageBackend>,
16    /// Pre-acquired read view held for the lifetime of this struct.
17    /// Using Arc allows sharing a single read view across multiple BackendTrieDB
18    /// instances (e.g., state trie + storage trie in a single query).
19    read_view: Arc<dyn StorageReadView>,
20    /// Last flatkeyvalue path already generated
21    last_computed_flatkeyvalue: Nibbles,
22    nodes_table: &'static str,
23    fkv_table: &'static str,
24    /// Storage trie address prefix (for storage tries)
25    /// None for state tries, Some(address) for storage tries
26    address_prefix: Option<H256>,
27}
28
29impl BackendTrieDB {
30    /// Create a new BackendTrieDB for the account trie
31    pub fn new_for_accounts(
32        db: Arc<dyn StorageBackend>,
33        last_written: Vec<u8>,
34    ) -> Result<Self, StoreError> {
35        let read_view = db.begin_read()?;
36        Self::new_for_accounts_with_view(db, read_view, last_written)
37    }
38
39    /// Create a new BackendTrieDB for the account trie with a shared read view
40    pub fn new_for_accounts_with_view(
41        db: Arc<dyn StorageBackend>,
42        read_view: Arc<dyn StorageReadView>,
43        last_written: Vec<u8>,
44    ) -> Result<Self, StoreError> {
45        let last_computed_flatkeyvalue = Nibbles::from_hex(last_written);
46        Ok(Self {
47            db,
48            read_view,
49            last_computed_flatkeyvalue,
50            nodes_table: ACCOUNT_TRIE_NODES,
51            fkv_table: ACCOUNT_FLATKEYVALUE,
52            address_prefix: None,
53        })
54    }
55
56    /// Create a new BackendTrieDB for the storage tries
57    pub fn new_for_storages(
58        db: Arc<dyn StorageBackend>,
59        last_written: Vec<u8>,
60    ) -> Result<Self, StoreError> {
61        let read_view = db.begin_read()?;
62        Self::new_for_storages_with_view(db, read_view, last_written)
63    }
64
65    /// Create a new BackendTrieDB for the storage tries with a shared read view
66    pub fn new_for_storages_with_view(
67        db: Arc<dyn StorageBackend>,
68        read_view: Arc<dyn StorageReadView>,
69        last_written: Vec<u8>,
70    ) -> Result<Self, StoreError> {
71        let last_computed_flatkeyvalue = Nibbles::from_hex(last_written);
72        Ok(Self {
73            db,
74            read_view,
75            last_computed_flatkeyvalue,
76            nodes_table: STORAGE_TRIE_NODES,
77            fkv_table: STORAGE_FLATKEYVALUE,
78            address_prefix: None,
79        })
80    }
81
82    /// Create a new BackendTrieDB for a specific storage trie
83    pub fn new_for_account_storage(
84        db: Arc<dyn StorageBackend>,
85        address_prefix: H256,
86        last_written: Vec<u8>,
87    ) -> Result<Self, StoreError> {
88        let read_view = db.begin_read()?;
89        Self::new_for_account_storage_with_view(db, read_view, address_prefix, last_written)
90    }
91
92    /// Create a new BackendTrieDB for a specific storage trie with a shared read view
93    pub fn new_for_account_storage_with_view(
94        db: Arc<dyn StorageBackend>,
95        read_view: Arc<dyn StorageReadView>,
96        address_prefix: H256,
97        last_written: Vec<u8>,
98    ) -> Result<Self, StoreError> {
99        let last_computed_flatkeyvalue = Nibbles::from_hex(last_written);
100        Ok(Self {
101            db,
102            read_view,
103            last_computed_flatkeyvalue,
104            nodes_table: STORAGE_TRIE_NODES,
105            fkv_table: STORAGE_FLATKEYVALUE,
106            address_prefix: Some(address_prefix),
107        })
108    }
109
110    fn make_key(&self, path: Nibbles) -> Vec<u8> {
111        apply_prefix(self.address_prefix, path).into_vec()
112    }
113
114    /// Key might be for an account or storage slot
115    fn table_for_key(&self, key: &[u8]) -> &'static str {
116        let is_leaf = key.len() == 65 || key.len() == 131;
117        if is_leaf {
118            self.fkv_table
119        } else {
120            self.nodes_table
121        }
122    }
123}
124
125impl TrieDB for BackendTrieDB {
126    fn flatkeyvalue_computed(&self, key: Nibbles) -> bool {
127        let key = apply_prefix(self.address_prefix, key);
128        self.last_computed_flatkeyvalue >= key
129    }
130
131    fn get(&self, key: Nibbles) -> Result<Option<Vec<u8>>, TrieError> {
132        let prefixed_key = self.make_key(key);
133        let table = self.table_for_key(&prefixed_key);
134        self.read_view
135            .get(table, prefixed_key.as_ref())
136            .map_err(|e| TrieError::DbError(anyhow::anyhow!("Failed to get from database: {}", e)))
137    }
138
139    fn put_batch(&self, key_values: Vec<(Nibbles, Vec<u8>)>) -> Result<(), TrieError> {
140        let mut tx = self.db.begin_write().map_err(|e| {
141            TrieError::DbError(anyhow::anyhow!("Failed to begin write transaction: {}", e))
142        })?;
143        for (key, value) in key_values {
144            let prefixed_key = self.make_key(key);
145            let table = self.table_for_key(&prefixed_key);
146            tx.put_batch(table, vec![(prefixed_key, value)])
147                .map_err(|e| TrieError::DbError(anyhow::anyhow!("Failed to write batch: {}", e)))?;
148        }
149        tx.commit()
150            .map_err(|e| TrieError::DbError(anyhow::anyhow!("Failed to write batch: {}", e)))
151    }
152}
153
154/// Read-only version with persistent locked transaction/snapshot for batch reads
155pub struct BackendTrieDBLocked {
156    account_trie_tx: Box<dyn StorageLockedView>,
157    storage_trie_tx: Box<dyn StorageLockedView>,
158    account_fkv_tx: Box<dyn StorageLockedView>,
159    storage_fkv_tx: Box<dyn StorageLockedView>,
160    /// Last flatkeyvalue path already generated
161    last_computed_flatkeyvalue: Nibbles,
162}
163
164impl BackendTrieDBLocked {
165    pub fn new(engine: &dyn StorageBackend, last_written: Vec<u8>) -> Result<Self, StoreError> {
166        let last_computed_flatkeyvalue = Nibbles::from_hex(last_written);
167        let account_trie_tx = engine.begin_locked(ACCOUNT_TRIE_NODES)?;
168        let storage_trie_tx = engine.begin_locked(STORAGE_TRIE_NODES)?;
169        let account_fkv_tx = engine.begin_locked(ACCOUNT_FLATKEYVALUE)?;
170        let storage_fkv_tx = engine.begin_locked(STORAGE_FLATKEYVALUE)?;
171        Ok(Self {
172            account_trie_tx,
173            storage_trie_tx,
174            account_fkv_tx,
175            storage_fkv_tx,
176            last_computed_flatkeyvalue,
177        })
178    }
179
180    /// Key is already prefixed
181    fn tx_for_key(&self, key: &Nibbles) -> &dyn StorageLockedView {
182        let is_leaf = key.len() == 65 || key.len() == 131;
183        let is_account = key.len() <= 65;
184        if is_leaf {
185            if is_account {
186                &*self.account_fkv_tx
187            } else {
188                &*self.storage_fkv_tx
189            }
190        } else if is_account {
191            &*self.account_trie_tx
192        } else {
193            &*self.storage_trie_tx
194        }
195    }
196}
197
198impl TrieDB for BackendTrieDBLocked {
199    fn flatkeyvalue_computed(&self, key: Nibbles) -> bool {
200        self.last_computed_flatkeyvalue >= key
201    }
202
203    fn get(&self, key: Nibbles) -> Result<Option<Vec<u8>>, TrieError> {
204        let tx = self.tx_for_key(&key);
205        tx.get(key.as_ref())
206            .map_err(|e| TrieError::DbError(anyhow::anyhow!("Failed to get from database: {}", e)))
207    }
208
209    fn put_batch(&self, _key_values: Vec<(Nibbles, Vec<u8>)>) -> Result<(), TrieError> {
210        // Read-only locked storage, should not be used for puts
211        Err(TrieError::DbError(anyhow::anyhow!("trie is read-only")))
212    }
213}