-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathmod.rs
339 lines (307 loc) · 12.8 KB
/
mod.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
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
use crate::kv::{
traits::{DefaultFlags, Mode, Table},
EnvFlags, MdbxCursor, MdbxEnv, MdbxTx,
};
use ethereum_types::{Address, H256, U256};
use eyre::{eyre, Result};
use mdbx::{TransactionKind, RO, RW};
mod macros;
pub mod models;
pub mod tables;
mod utils;
use utils::consts as C;
use models::*;
use tables::*;
pub const NUM_TABLES: usize = 50;
// https://github.com/ledgerwatch/erigon-lib/blob/625c9f5385d209dc2abfadedf6e4b3914a26ed3e/kv/mdbx/kv_mdbx.go#L154
pub const ENV_FLAGS: EnvFlags = EnvFlags {
no_rdahead: true,
coalesce: true,
accede: true,
no_sub_dir: false,
exclusive: false,
no_meminit: false,
liforeclaim: false,
};
/// Open an mdbx env with Erigon-specific configuration.
pub fn env_open<M: Mode>(path: &std::path::Path) -> Result<MdbxEnv<M>> {
MdbxEnv::<M>::open(path, NUM_TABLES, ENV_FLAGS)
}
/// Erigon wraps an `MdbxTx` and provides Erigon-specific access methods.
pub struct Erigon<'env, K: TransactionKind>(pub MdbxTx<'env, K>);
impl<'env> Erigon<'env, RO> {
pub fn begin(env: &'env MdbxEnv<RO>) -> Result<Self> {
env.begin().map(Self)
}
}
impl<'env> Erigon<'env, RW> {
pub fn begin_rw(env: &'env MdbxEnv<RW>) -> Result<Self> {
env.begin_rw().map(Self)
}
}
impl<'env, K: TransactionKind> Erigon<'env, K> {
pub fn new(inner: MdbxTx<'env, K>) -> Self {
Self(inner)
}
}
impl<'env, K: Mode> Erigon<'env, K> {
/// Opens and reads from the db table with the table's default flags
pub fn read<'tx, T>(&'tx self, key: T::Key) -> Result<Option<T::Value>>
where
T: Table<'tx> + DefaultFlags,
{
self.0.get::<T, T::Flags>(self.0.open_db()?, key)
}
/// Opens a table with the table's default flags and creates a cursor into
/// the opened table.
pub fn cursor<'tx, T>(&'tx self) -> Result<MdbxCursor<'tx, K, T>>
where
T: Table<'tx> + DefaultFlags,
{
self.0.cursor::<T, T::Flags>(self.0.open_db()?)
}
/// Returns the hash of the current canonical head header.
pub fn read_head_header_hash(&self) -> Result<Option<H256>> {
self.read::<LastHeader>(LastHeaderKey)
}
/// Returns the hash of the current canonical head block.
pub fn read_head_block_hash(&self) -> Result<Option<H256>> {
self.read::<LastBlock>(LastBlockKey)
}
/// Returns the incarnation of the account when it was last deleted.
pub fn read_incarnation(&self, adr: Address) -> Result<Option<Incarnation>> {
self.read::<IncarnationMap>(adr)
}
/// Returns the decoded account data as stored in the PlainState table.
pub fn read_account(&self, adr: Address) -> Result<Option<Account>> {
self.read::<PlainState>(adr)
}
/// Returns the number of the block containing the specified transaction.
pub fn read_transaction_block_number(&self, hash: H256) -> Result<Option<U256>> {
self.read::<BlockTransactionLookup>(hash)
}
/// Returns the block header identified by the (block number, block hash) key
pub fn read_header(&self, key: impl Into<HeaderKey>) -> Result<Option<BlockHeader>> {
self.read::<Header>(key.into())
}
/// Returns header total difficulty
pub fn read_total_difficulty(
&self,
key: impl Into<HeaderKey>,
) -> Result<Option<TotalDifficulty>> {
self.read::<HeadersTotalDifficulty>(key.into())
}
/// Returns the decoding of the body as stored in the BlockBody table
pub fn read_body_for_storage(
&self,
key: impl Into<HeaderKey>,
) -> Result<Option<BodyForStorage>> {
let key = key.into();
self.read::<BlockBody>(key)?
.map(|mut body| {
// Skip 1 system tx at the beginning of the block and 1 at the end
// https://github.com/ledgerwatch/erigon/blob/f56d4c5881822e70f65927ade76ef05bfacb1df4/core/rawdb/accessors_chain.go#L602-L605
// https://github.com/ledgerwatch/erigon-lib/blob/625c9f5385d209dc2abfadedf6e4b3914a26ed3e/kv/tables.go#L28
body.base_tx_id += 1;
body.tx_amount = body.tx_amount.checked_sub(2).ok_or_else(|| {
eyre!(
"Block body has too few txs: {}. HeaderKey: {:?}",
body.tx_amount,
key,
)
})?;
Ok(body)
})
.transpose()
}
/// Returns the header number assigned to a hash.
pub fn read_header_number(&self, hash: H256) -> Result<Option<BlockNumber>> {
self.read::<HeaderNumber>(hash)
}
/// Returns the number of the current canonical block header.
pub fn read_head_block_number(&self) -> Result<Option<BlockNumber>> {
let hash = self.read_head_header_hash()?.ok_or(eyre!("No value"))?;
self.read_header_number(hash)
}
/// Returns the signers of each transaction in the block.
pub fn read_senders(&self, key: impl Into<HeaderKey>) -> Result<Option<Vec<Address>>> {
self.read::<TxSender>(key.into())
}
/// Returns the hash assigned to a canonical block number.
pub fn read_canonical_hash(&self, num: impl Into<BlockNumber>) -> Result<Option<H256>> {
self.read::<CanonicalHeader>(num.into())
}
/// Determines whether a header with the given hash is on the canonical chain.
pub fn is_canonical_hash(&self, hash: H256) -> Result<bool> {
let num = self.read_header_number(hash)?.ok_or(eyre!("No value"))?;
let canon = self.read_canonical_hash(num)?.ok_or(eyre!("No value"))?;
Ok(canon != Default::default() && canon == hash)
}
/// Returns the value of the storage for account `adr` indexed by `slot`.
pub fn read_storage(
&self,
adr: Address,
inc: impl Into<Incarnation>,
slot: H256,
) -> Result<Option<U256>> {
let bucket = StorageKey(adr, inc.into());
let mut cur = self.cursor::<Storage>()?;
cur.seek_dup(bucket, slot)
.map(|kv| kv.and_then(|(k, v)| if k == slot { Some(v) } else { None }))
}
/// Returns an iterator over all of the storage (key, value) pairs for the
/// given address and account incarnation. If a start_slot is provided, the
/// iterator will begin at the smallest slot >= start_slot.
pub fn walk_storage(
&self,
adr: Address,
inc: impl Into<Incarnation>,
start_slot: Option<H256>,
) -> Result<impl Iterator<Item = Result<(H256, U256)>>> {
let key = StorageKey(adr, inc.into());
self.cursor::<Storage>()?.walk_dup(key, start_slot.unwrap_or_default())
}
/// Returns the code associated with the given codehash.
pub fn read_code(&self, codehash: H256) -> Result<Option<Bytecode>> {
if codehash == C::EMPTY_HASH {
return Ok(Default::default());
}
self.read::<Code>(codehash)
}
/// Returns the codehash at the `adr` with incarnation `inc`
pub fn read_codehash(&self, adr: Address, inc: impl Into<Incarnation>) -> Result<Option<H256>> {
let key = PlainCodeKey(adr, inc.into());
self.read::<PlainCodeHash>(key)
}
pub fn walk_txs_canonical(
&self,
start_key: Option<TxIndex>,
) -> Result<impl Iterator<Item = Result<(TxIndex, Transaction)>>> {
self.cursor::<BlockTransaction>()?
.walk(start_key.unwrap_or_default())
}
pub fn walk_txs_noncanonical(
&self,
start_key: Option<TxIndex>,
) -> Result<impl Iterator<Item = Result<(TxIndex, Transaction)>>> {
self.cursor::<NonCanonicalTransaction>()?
.walk(start_key.unwrap_or_default())
}
// The `AccountChangeSet` table at block `N` stores the state of all accounts
// changed in block `N` *before* block `N` changed them.
//
// The state of an account *after* the most recent change is always stored in the `PlainState` table.
//
// If Account A was changed in block 5 and again in block 25, the state of A for any
// block `[5, 25)` is stored in the `AccountChangeSet` table addressed by the
// block number 25. If we want to find the state of account `A` at block `B`,
// we first use the `AccountHistory` table to figure out which block to look for
// in the `AccountChangeSet` table. That is, we look for the smallest
// block >= `B` in which account `A` was changed, then we lookup the state
// of account `A` immediately before that change in the `AccountChangeSet` table.
//
// The `AccountHistory` table stores a roaring bitmap of the block numbers
// in which account `A` was changed. We search the bitmap for the smallest
// block number it contains which is `>= B`, then we read the state of account
// `A` at this block from the `AccountChangeSet` table.
//
// The confusing thing is, the block number in `AccountHistory` seems to
// be basically unused. For account `A`, every time a change is made, the
// bitmap stored at key `(A, u64::MAX)` is updated. Presumably this is used to
// grow the bitmap, and that's why akula and erigon both do some crazy mapping
// over the bitmap tables
//
// Notes:
// - `AccountHistory` and `StorageHistory` are written [here](https://github.com/ledgerwatch/erigon/blob/f9d7cb5ca9e8a135a76ddcb6fa4ee526ea383554/core/state/db_state_writer.go#L179).
// - `GetAsOf()` Erigon implementation [here](https://github.com/ledgerwatch/erigon/blob/f9d7cb5ca9e8a135a76ddcb6fa4ee526ea383554/core/state/history.go#L19).
//
/// Returns the state of account `adr` at the given block number.
pub fn read_account_hist(
&self,
adr: Address,
block: impl Into<BlockNumber>,
) -> Result<Option<Account>> {
let block = block.into();
let mut hist_cur = self.cursor::<AccountHistory>()?;
let (_, bitmap) = hist_cur
.seek((adr, block).into())?
.ok_or(eyre!("No value"))?;
let cs_block = match utils::find_gte(bitmap, *block) {
Some(changeset) => BlockNumber(changeset),
_ => return Ok(None),
};
let mut cs_cur = self.cursor::<AccountChangeSet>()?;
if let Some(AccountCSVal(k, mut acct)) = cs_cur.seek_dup(cs_block, adr)? {
if k == adr {
// recover the codehash
if *acct.incarnation > 0 && acct.codehash == Default::default() {
acct.codehash = self
.read_codehash(adr, acct.incarnation)?
.ok_or(eyre!("No value"))?
}
return Ok(Some(acct));
}
}
Ok(None)
}
/// Returns the value of an address's storage at the given block number. Returns `None` if the state
/// is not found in history (e.g., if it's in the PlainState table instead).
pub fn read_storage_hist(
&self,
adr: Address,
inc: impl Into<Incarnation>,
slot: H256,
block: impl Into<BlockNumber>,
) -> Result<Option<U256>> {
let block = block.into();
let mut hist_cur = self.cursor::<StorageHistory>()?;
let (_, bitmap) = hist_cur
.seek((adr, slot, block).into())?
.ok_or(eyre!("No value"))?;
let cs_block = match utils::find_gte(bitmap, *block) {
Some(changeset) => BlockNumber(changeset),
_ => return Ok(None),
};
let cs_key = (cs_block, adr, inc.into()).into();
let mut cs_cur = self.cursor::<StorageChangeSet>()?;
if let Some(StorageCSVal(k, v)) = cs_cur.seek_dup(cs_key, slot)? {
if k == slot {
return Ok(Some(v));
}
}
Ok(None)
}
}
impl<'env> Erigon<'env, mdbx::RW> {
/// Opens and writes to the db table with the table's default flags.
pub fn write<'tx, T>(&'tx self, key: T::Key, val: T::Value) -> Result<()>
where
T: Table<'tx> + DefaultFlags,
{
self.0.put::<T, T::Flags>(self.0.open_db()?, key, val)
}
pub fn write_head_header_hash(&self, v: H256) -> Result<()> {
self.write::<LastHeader>(LastHeaderKey, v)
}
pub fn write_head_block_hash(&self, v: H256) -> Result<()> {
self.write::<LastBlock>(LastBlockKey, v)
}
pub fn write_incarnation(&self, k: Address, v: Incarnation) -> Result<()> {
self.write::<IncarnationMap>(k, v)
}
pub fn write_account(&self, k: Address, v: Account) -> Result<()> {
self.write::<PlainState>(k, v)
}
pub fn write_transaction_block_number(&self, k: H256, v: U256) -> Result<()> {
self.write::<BlockTransactionLookup>(k, v)
}
pub fn write_header_number(&self, k: H256, v: BlockNumber) -> Result<()> {
self.write::<HeaderNumber>(k, v)
}
pub fn write_header(&self, k: HeaderKey, v: BlockHeader) -> Result<()> {
self.write::<Header>(k, v)
}
pub fn write_body_for_storage(&self, k: HeaderKey, v: BodyForStorage) -> Result<()> {
self.write::<BlockBody>(k, v)
}
}