2023-11-01 10:32:09 +07:00
|
|
|
# Nimbus
|
2024-02-21 23:04:59 +07:00
|
|
|
# Copyright (c) 2020-2024 Status Research & Development GmbH
|
2023-11-01 10:32:09 +07:00
|
|
|
# Licensed under either of
|
|
|
|
# * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE) or
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0)
|
|
|
|
# * MIT license ([LICENSE-MIT](LICENSE-MIT) or
|
|
|
|
# http://opensource.org/licenses/MIT)
|
|
|
|
# at your option. This file may not be copied, modified, or distributed except
|
|
|
|
# according to those terms.
|
|
|
|
|
2020-05-06 12:34:54 +07:00
|
|
|
import
|
|
|
|
randutils, random, parseopt, strutils, os,
|
|
|
|
eth/[common, rlp], eth/trie/[hexary, db, trie_defs],
|
|
|
|
nimcrypto/sysrand, ../stateless/[json_from_tree],
|
2023-03-10 13:42:37 -05:00
|
|
|
../nimbus/db/[storage_types, distinct_tries],
|
|
|
|
./witness_types, ./multi_keys
|
2020-05-06 12:34:54 +07:00
|
|
|
|
|
|
|
type
|
|
|
|
DB = TrieDatabaseRef
|
|
|
|
|
2024-02-21 23:04:59 +07:00
|
|
|
StorageKeys = tuple[storageRoot: Hash256, keys: MultiKeysRef]
|
2020-05-06 12:34:54 +07:00
|
|
|
|
|
|
|
AccountDef = object
|
|
|
|
storageKeys: MultiKeysRef
|
|
|
|
account: Account
|
|
|
|
codeTouched: bool
|
|
|
|
|
|
|
|
proc randU256(): UInt256 =
|
|
|
|
var bytes: array[32, byte]
|
|
|
|
discard randomBytes(bytes[0].addr, sizeof(result))
|
|
|
|
result = UInt256.fromBytesBE(bytes)
|
|
|
|
|
|
|
|
proc randStorageSlot(): StorageSlot =
|
|
|
|
discard randomBytes(result[0].addr, sizeof(result))
|
|
|
|
|
|
|
|
proc randNonce(): AccountNonce =
|
|
|
|
discard randomBytes(result.addr, sizeof(result))
|
|
|
|
|
|
|
|
proc randCode(db: DB): Hash256 =
|
|
|
|
if rand(0..1) == 0:
|
|
|
|
result = blankStringHash
|
|
|
|
else:
|
|
|
|
let codeLen = rand(1..150)
|
|
|
|
let code = randList(byte, rng(0, 255), codeLen, unique = false)
|
2022-09-03 20:15:35 +02:00
|
|
|
result = keccakHash(code)
|
2020-05-06 12:34:54 +07:00
|
|
|
db.put(contractHashKey(result).toOpenArray, code)
|
|
|
|
|
|
|
|
proc randStorage(db: DB, numSlots: int): StorageKeys =
|
|
|
|
if rand(0..1) == 0 or numSlots == 0:
|
2024-02-21 23:04:59 +07:00
|
|
|
result = (emptyRlpHash, MultiKeysRef(nil))
|
2020-05-06 12:34:54 +07:00
|
|
|
else:
|
2023-03-10 13:42:37 -05:00
|
|
|
var trie = initStorageTrie(db)
|
2020-05-06 12:34:54 +07:00
|
|
|
var keys = newSeq[StorageSlot](numSlots)
|
|
|
|
|
|
|
|
for i in 0..<numSlots:
|
|
|
|
keys[i] = randStorageSlot()
|
2023-03-10 13:42:37 -05:00
|
|
|
trie.putSlotBytes(keys[i], rlp.encode(randU256()))
|
2020-05-06 12:34:54 +07:00
|
|
|
|
|
|
|
if rand(0..1) == 0:
|
2024-02-21 23:04:59 +07:00
|
|
|
result = (trie.rootHash, MultiKeysRef(nil))
|
2020-05-06 12:34:54 +07:00
|
|
|
else:
|
|
|
|
var m = newMultikeys(keys)
|
|
|
|
result = (trie.rootHash, m)
|
|
|
|
|
|
|
|
proc randAccount(db: DB, numSlots: int): AccountDef =
|
|
|
|
result.account.nonce = randNonce()
|
|
|
|
result.account.balance = randU256()
|
|
|
|
let z = randStorage(db, numSlots)
|
|
|
|
result.account.codeHash = randCode(db)
|
|
|
|
result.account.storageRoot = z.storageRoot
|
|
|
|
result.storageKeys = z.keys
|
|
|
|
result.codeTouched = rand(0..1) == 0
|
|
|
|
|
|
|
|
proc randAddress(): EthAddress =
|
|
|
|
discard randomBytes(result.addr, sizeof(result))
|
|
|
|
|
|
|
|
proc runGenerator(numPairs, numSlots: int): string =
|
|
|
|
var memDB = newMemoryDB()
|
2023-03-10 13:42:37 -05:00
|
|
|
var trie = initAccountsTrie(memDB)
|
2020-05-06 12:34:54 +07:00
|
|
|
var addrs = newSeq[AccountKey](numPairs)
|
|
|
|
var accs = newSeq[Account](numPairs)
|
|
|
|
|
|
|
|
for i in 0..<numPairs:
|
|
|
|
let acc = randAccount(memDB, numSlots)
|
|
|
|
addrs[i] = (randAddress(), acc.codeTouched, acc.storageKeys)
|
|
|
|
accs[i] = acc.account
|
2023-03-10 13:42:37 -05:00
|
|
|
trie.putAccountBytes(addrs[i].address, rlp.encode(accs[i]))
|
2020-05-06 12:34:54 +07:00
|
|
|
|
|
|
|
var mkeys = newMultiKeys(addrs)
|
|
|
|
let rootHash = trie.rootHash
|
|
|
|
|
|
|
|
var wb = initWitnessBuilder(memDB, rootHash, {wfEIP170})
|
|
|
|
result = wb.buildWitness(mkeys)
|
|
|
|
|
|
|
|
proc writeHelp() =
|
2020-05-07 22:12:34 +07:00
|
|
|
echo "json_witness_gen output --pairs:val --slots:val -s:val -p:val"
|
2020-05-06 12:34:54 +07:00
|
|
|
|
|
|
|
proc main() =
|
|
|
|
var filename: string
|
|
|
|
var outputDir: string
|
|
|
|
var numPairs = 1
|
|
|
|
var numSlots = 1
|
|
|
|
for kind, key, val in getopt():
|
|
|
|
case kind
|
|
|
|
of cmdArgument:
|
|
|
|
filename = key
|
|
|
|
of cmdLongOption, cmdShortOption:
|
|
|
|
case key
|
|
|
|
of "pairs", "p":
|
|
|
|
numPairs = parseInt(val)
|
|
|
|
if numPairs <= 0: numPairs = 1
|
|
|
|
of "slots", "s":
|
|
|
|
numSlots = parseInt(val)
|
|
|
|
if numSlots < 0: numSlots = 0
|
|
|
|
of "output", "o":
|
|
|
|
outputDir = val
|
|
|
|
of cmdEnd: assert(false) # cannot happen
|
|
|
|
|
|
|
|
if filename == "":
|
|
|
|
writeHelp()
|
|
|
|
quit(0)
|
|
|
|
|
|
|
|
randomize()
|
|
|
|
let witness = runGenerator(numPairs, numSlots)
|
|
|
|
let filePath = if outputDir.len > 0: outputDir / filename: else: filename
|
|
|
|
writeFile(filePath, witness)
|
|
|
|
|
|
|
|
main()
|