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
11pub struct BackendTrieDB {
14 db: Arc<dyn StorageBackend>,
16 read_view: Arc<dyn StorageReadView>,
20 last_computed_flatkeyvalue: Nibbles,
22 nodes_table: &'static str,
23 fkv_table: &'static str,
24 address_prefix: Option<H256>,
27}
28
29impl BackendTrieDB {
30 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 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 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 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 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 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 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
154pub 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_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 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 Err(TrieError::DbError(anyhow::anyhow!("trie is read-only")))
212 }
213}