2024-05-16 08:16:16 +00:00
|
|
|
# Fluffy
|
|
|
|
# Copyright (c) 2021-2024 Status Research & Development GmbH
|
|
|
|
# Licensed and distributed under either of
|
|
|
|
# * MIT license (license terms in the root directory or at https://opensource.org/licenses/MIT).
|
|
|
|
# * Apache v2 license (license terms in the root directory or at https://www.apache.org/licenses/LICENSE-2.0).
|
|
|
|
# at your option. This file may not be copied, modified, or distributed except according to those terms.
|
|
|
|
|
|
|
|
{.push raises: [].}
|
|
|
|
|
|
|
|
import
|
|
|
|
std/[sugar, sequtils],
|
|
|
|
eth/[common, trie, trie/db],
|
|
|
|
../../nimbus/common/chain_config,
|
2024-05-28 00:45:30 +00:00
|
|
|
../../network/state/[state_content, state_utils],
|
|
|
|
../../eth_data/yaml_utils
|
|
|
|
|
|
|
|
export yaml_utils
|
|
|
|
|
|
|
|
const testVectorDir* = "./vendor/portal-spec-tests/tests/mainnet/state/validation/"
|
|
|
|
|
|
|
|
type
|
|
|
|
YamlTrieNodeRecursiveGossipKV* = ref object
|
|
|
|
content_key*: string
|
|
|
|
content_value_offer*: string
|
|
|
|
content_value_retrieval*: string
|
|
|
|
|
|
|
|
YamlTrieNodeKV* = object
|
|
|
|
state_root*: string
|
|
|
|
content_key*: string
|
|
|
|
content_value_offer*: string
|
|
|
|
content_value_retrieval*: string
|
|
|
|
recursive_gossip*: YamlTrieNodeRecursiveGossipKV
|
|
|
|
|
|
|
|
YamlTrieNodeKVs* = seq[YamlTrieNodeKV]
|
|
|
|
|
|
|
|
YamlContractBytecodeKV* = object
|
|
|
|
state_root*: string
|
|
|
|
content_key*: string
|
|
|
|
content_value_offer*: string
|
|
|
|
content_value_retrieval*: string
|
|
|
|
|
|
|
|
YamlContractBytecodeKVs* = seq[YamlContractBytecodeKV]
|
|
|
|
|
|
|
|
YamlRecursiveGossipKV* = object
|
|
|
|
content_key*: string
|
|
|
|
content_value*: string
|
|
|
|
|
|
|
|
YamlRecursiveGossipData* = object
|
|
|
|
state_root*: string
|
|
|
|
recursive_gossip*: seq[YamlRecursiveGossipKV]
|
|
|
|
|
|
|
|
YamlRecursiveGossipKVs* = seq[YamlRecursiveGossipData]
|
2024-05-16 08:16:16 +00:00
|
|
|
|
2024-05-24 07:49:51 +00:00
|
|
|
func asNibbles*(key: openArray[byte], isEven = true): Nibbles =
|
2024-05-16 08:16:16 +00:00
|
|
|
Nibbles.init(key, isEven)
|
|
|
|
|
2024-05-24 07:49:51 +00:00
|
|
|
func removeLeafKeyEndNibbles*(
|
|
|
|
nibbles: Nibbles, leafNode: TrieNode
|
|
|
|
): Nibbles {.raises: [RlpError].} =
|
|
|
|
let nodeRlp = rlpFromBytes(leafNode.asSeq())
|
|
|
|
doAssert(nodeRlp.listLen() == 2)
|
|
|
|
let (_, isLeaf, prefix) = decodePrefix(nodeRlp.listElem(0))
|
|
|
|
doAssert(isLeaf)
|
|
|
|
|
|
|
|
let leafPrefix = prefix.unpackNibbles()
|
|
|
|
var unpackedNibbles = nibbles.unpackNibbles()
|
|
|
|
doAssert(unpackedNibbles[^leafPrefix.len() .. ^1] == leafPrefix)
|
|
|
|
|
2024-05-28 00:45:30 +00:00
|
|
|
unpackedNibbles.dropN(leafPrefix.len()).packNibbles()
|
2024-05-24 07:49:51 +00:00
|
|
|
|
|
|
|
func asTrieProof*(branch: openArray[seq[byte]]): TrieProof =
|
2024-05-16 08:16:16 +00:00
|
|
|
TrieProof.init(branch.map(node => TrieNode.init(node)))
|
|
|
|
|
|
|
|
proc getTrieProof*(
|
|
|
|
state: HexaryTrie, key: openArray[byte]
|
|
|
|
): TrieProof {.raises: [RlpError].} =
|
|
|
|
let branch = state.getBranch(key)
|
2024-05-28 00:45:30 +00:00
|
|
|
# for node in branch:
|
|
|
|
# debugEcho rlp.decode(node)
|
2024-05-16 08:16:16 +00:00
|
|
|
branch.asTrieProof()
|
|
|
|
|
|
|
|
proc generateAccountProof*(
|
|
|
|
state: HexaryTrie, address: EthAddress
|
|
|
|
): TrieProof {.raises: [RlpError].} =
|
|
|
|
let key = keccakHash(address).data
|
|
|
|
state.getTrieProof(key)
|
|
|
|
|
|
|
|
proc generateStorageProof*(
|
|
|
|
state: HexaryTrie, slotKey: UInt256
|
|
|
|
): TrieProof {.raises: [RlpError].} =
|
|
|
|
let key = keccakHash(toBytesBE(slotKey)).data
|
|
|
|
state.getTrieProof(key)
|
|
|
|
|
|
|
|
proc getGenesisAlloc*(filePath: string): GenesisAlloc =
|
|
|
|
var cn: NetworkParams
|
|
|
|
if not loadNetworkParams(filePath, cn):
|
|
|
|
quit(1)
|
|
|
|
|
|
|
|
cn.genesis.alloc
|
|
|
|
|
|
|
|
proc toState*(
|
|
|
|
alloc: GenesisAlloc
|
|
|
|
): (HexaryTrie, Table[EthAddress, HexaryTrie]) {.raises: [RlpError].} =
|
|
|
|
var accountTrie = initHexaryTrie(newMemoryDB())
|
|
|
|
var storageStates = initTable[EthAddress, HexaryTrie]()
|
|
|
|
|
|
|
|
for address, genAccount in alloc:
|
|
|
|
var storageRoot = EMPTY_ROOT_HASH
|
|
|
|
var codeHash = EMPTY_CODE_HASH
|
|
|
|
|
|
|
|
if genAccount.code.len() > 0:
|
|
|
|
var storageTrie = initHexaryTrie(newMemoryDB())
|
|
|
|
for slotKey, slotValue in genAccount.storage:
|
|
|
|
let key = keccakHash(toBytesBE(slotKey)).data
|
|
|
|
let value = rlp.encode(slotValue)
|
|
|
|
storageTrie.put(key, value)
|
|
|
|
storageStates[address] = storageTrie
|
|
|
|
storageRoot = storageTrie.rootHash()
|
|
|
|
codeHash = keccakHash(genAccount.code)
|
|
|
|
|
|
|
|
let account = Account(
|
|
|
|
nonce: genAccount.nonce,
|
|
|
|
balance: genAccount.balance,
|
|
|
|
storageRoot: storageRoot,
|
|
|
|
codeHash: codeHash,
|
|
|
|
)
|
|
|
|
let key = keccakHash(address).data
|
|
|
|
let value = rlp.encode(account)
|
|
|
|
accountTrie.put(key, value)
|
|
|
|
|
|
|
|
(accountTrie, storageStates)
|