Remove hunter (#2697)

* Remove `hunter`

why:
  Neither functional anymore, nor used

* Remove obsolete premix

* Remove obsolete launcher

---------

Co-authored-by: jangko <jangko128@gmail.com>
This commit is contained in:
Jordan Hrycaj 2024-10-06 10:11:44 +00:00 committed by GitHub
parent 6565544d35
commit 3822c57ddc
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
35 changed files with 31 additions and 2200 deletions

View File

@ -60,13 +60,10 @@ EXCLUDED_NIM_PACKAGES := \
# debugging tools + testing tools # debugging tools + testing tools
TOOLS := \ TOOLS := \
test_tools_build \ test_tools_build \
persist \
hunter \
nrpc nrpc
TOOLS_DIRS := \ TOOLS_DIRS := \
nrpc \ nrpc \
tests \ tests
premix
# comma-separated values for the "clean" target # comma-separated values for the "clean" target
TOOLS_CSV := $(subst $(SPACE),$(COMMA),$(TOOLS)) TOOLS_CSV := $(subst $(SPACE),$(COMMA),$(TOOLS))

View File

@ -21,7 +21,7 @@ NIM_FLAGS="c -d:release"
echo "## ${1}" > simulators.md echo "## ${1}" > simulators.md
# more suites: engine, graphql, rpc # more suites: graphql, rpc
suites=(consensus pyspec engine) suites=(consensus pyspec engine)
for suite in "${suites[@]}" for suite in "${suites[@]}"
do do

View File

@ -9,9 +9,11 @@
# according to those terms. # according to those terms.
import import
eth/common/eth_types, eth/common,
eth/common/hashes,
stint, stint,
kzg4844/kzg, kzg4844/kzg,
kzg4844/kzg_abi,
stew/endians2, stew/endians2,
nimcrypto/sha2, nimcrypto/sha2,
results, results,
@ -138,9 +140,9 @@ proc generateBlob(blobid: BlobID): BlobCommitment =
doAssert(false, res.error) doAssert(false, res.error)
result.commitment = res.get result.commitment = res.get
proc getVersionedHash*(blobid: BlobID, commitmentVersion: byte): Hash256 = proc getVersionedHash*(blobid: BlobID, commitmentVersion: byte): Hash32 =
let res = blobid.generateBlob() let res = blobid.generateBlob()
result = sha256.digest(res.commitment.bytes) result = Hash32 sha256.digest(res.commitment.bytes).data
result.data[0] = commitmentVersion result.data[0] = commitmentVersion
proc blobDataGenerator*(startBlobId: BlobID, blobCount: int): BlobTxWrapData = proc blobDataGenerator*(startBlobId: BlobID, blobCount: int): BlobTxWrapData =

View File

@ -126,7 +126,7 @@ method getPayloadAttributes(cust: BasePayloadAttributesCustomizer, basePayloadAt
customPayloadAttributes.timestamp = w3Qty cust.timestamp.get customPayloadAttributes.timestamp = w3Qty cust.timestamp.get
if cust.prevRandao.isSome: if cust.prevRandao.isSome:
customPayloadAttributes.prevRandao = w3Hash cust.prevRandao.get customPayloadAttributes.prevRandao = cust.prevRandao.get
if cust.suggestedFeeRecipient.isSome: if cust.suggestedFeeRecipient.isSome:
customPayloadAttributes.suggestedFeeRecipient = w3Addr cust.suggestedFeeRecipient.get customPayloadAttributes.suggestedFeeRecipient = w3Addr cust.suggestedFeeRecipient.get
@ -338,7 +338,7 @@ func getTimestamp*(cust: CustomPayloadData, basePayload: ExecutionPayload): uint
proc customizePayload*(cust: CustomPayloadData, data: ExecutableData): ExecutableData {.gcsafe.} = proc customizePayload*(cust: CustomPayloadData, data: ExecutableData): ExecutableData {.gcsafe.} =
var customHeader = blockHeader(data.basePayload, beaconRoot = data.beaconRoot) var customHeader = blockHeader(data.basePayload, beaconRoot = data.beaconRoot)
if cust.transactions.isSome: if cust.transactions.isSome:
customHeader.txRoot = calcTxRoot(cust.transactions.get) customHeader.transactionsRoot = calcTxRoot(cust.transactions.get)
# Overwrite custom information # Overwrite custom information
if cust.parentHash.isSome: if cust.parentHash.isSome:
@ -533,13 +533,13 @@ type
ExtraVersionedHashes ExtraVersionedHashes
InvalidWithdrawals InvalidWithdrawals
func scramble(data: Web3Hash): Opt[common.Hash256] = func scramble(data: Web3Hash): Opt[Hash32] =
var h = ethHash data var h = ethHash data
h.data[^1] = byte(255 - h.data[^1]) h.data[^1] = byte(255 - h.data[^1])
Opt.some(h) Opt.some(h)
func scramble(data: common.Hash256): Opt[common.Hash256] = func scramble(data: Bytes32): Opt[Hash32] =
var h = data var h = Hash32 data
h.data[0] = byte(255 - h.data[0]) h.data[0] = byte(255 - h.data[0])
Opt.some(h) Opt.some(h)
@ -585,9 +585,9 @@ proc generateInvalidPayload*(sender: TxSender, data: ExecutableData, payloadFiel
of InvalidPrevRandao: of InvalidPrevRandao:
# This option potentially requires a transaction that uses the PREVRANDAO opcode. # This option potentially requires a transaction that uses the PREVRANDAO opcode.
# Otherwise the payload will still be valid. # Otherwise the payload will still be valid.
let randomHash = common.Hash256.randomBytes() let randomHash = common.Hash32.randomBytes()
customPayloadMod = CustomPayloadData( customPayloadMod = CustomPayloadData(
prevRandao: Opt.some(randomHash), prevRandao: Opt.some(Bytes32 randomHash.data),
) )
of InvalidParentBeaconBlockRoot: of InvalidParentBeaconBlockRoot:
doAssert(data.beaconRoot.isSome, doAssert(data.beaconRoot.isSome,

View File

@ -654,10 +654,10 @@ proc produceSingleBlock*(cl: CLMocker, cb: BlockProcessCallbacks): bool {.gcsafe
return false return false
# mixHash == prevRandao # mixHash == prevRandao
if newHeader.mixHash != cl.prevRandaoHistory[cl.latestHeadNumber]: if newHeader.mixHash != Bytes32 cl.prevRandaoHistory[cl.latestHeadNumber]:
error "CLMocker: Client produced a new header with incorrect mixHash", error "CLMocker: Client produced a new header with incorrect mixHash",
get = newHeader.mixHash.data.toHex, get = newHeader.mixHash,
expect = cl.prevRandaoHistory[cl.latestHeadNumber].data.toHex expect = cl.prevRandaoHistory[cl.latestHeadNumber]
return false return false
# nonce == 0x0000000000000000 # nonce == 0x0000000000000000

View File

@ -23,6 +23,7 @@ import
core/tx_pool/tx_item, core/tx_pool/tx_item,
core/block_import, core/block_import,
rpc, rpc,
sync/handlers,
beacon/beacon_engine, beacon/beacon_engine,
beacon/web3_eth_conv, beacon/web3_eth_conv,
common common

View File

@ -198,8 +198,8 @@ proc makeTxOfType(params: MakeTxParams, tc: BaseTx): PooledTransaction =
), ),
networkPayload: NetworkPayload( networkPayload: NetworkPayload(
blobs: blobData.blobs.mapIt(it.bytes), blobs: blobData.blobs.mapIt(it.bytes),
commitments: blobData.commitments.mapIt(it.bytes), commitments: blobData.commitments.mapIt(KzgCommitment it.bytes),
proofs: blobData.proofs.mapIt(it.bytes), proofs: blobData.proofs.mapIt(KzgProof it.bytes),
) )
) )
else: else:
@ -342,8 +342,8 @@ proc makeTx*(params: MakeTxParams, tc: BlobTx): PooledTransaction =
tx: signTransaction(unsignedTx, params.key), tx: signTransaction(unsignedTx, params.key),
networkPayload: NetworkPayload( networkPayload: NetworkPayload(
blobs : data.blobs.mapIt(it.bytes), blobs : data.blobs.mapIt(it.bytes),
commitments: data.commitments.mapIt(it.bytes), commitments: data.commitments.mapIt(KzgCommitment it.bytes),
proofs : data.proofs.mapIt(it.bytes), proofs : data.proofs.mapIt(KzgProof it.bytes),
), ),
) )

View File

@ -79,10 +79,10 @@ const ZeroAddr* = default(EthAddress)
func toHash*(x: UInt256): common.Hash256 = func toHash*(x: UInt256): common.Hash256 =
common.Hash32(x.toByteArrayBE) common.Hash32(x.toByteArrayBE)
func timestampToBeaconRoot*(timestamp: Quantity): Web3FixedBytes[32] = func timestampToBeaconRoot*(timestamp: Quantity): Hash32 =
# Generates a deterministic hash from the timestamp # Generates a deterministic hash from the timestamp
let h = sha2.sha256.digest(timestamp.uint64.toBytesBE) let h = sha2.sha256.digest(timestamp.uint64.toBytesBE)
Web3FixedBytes[32](h.data) Hash32(h.data)
proc randomBytes*(_: type common.Hash256): common.Hash256 = proc randomBytes*(_: type common.Hash256): common.Hash256 =
doAssert randomBytes(result.data) == 32 doAssert randomBytes(result.data) == 32

View File

@ -158,12 +158,6 @@ proc persistBlocksImpl(
skipUncles = NoPersistUncles in flags, skipUncles = NoPersistUncles in flags,
) )
# when defined(nimbusDumpDebuggingMetaData):
# if validationResult == ValidationResult.Error and
# body.transactions.calcTxRoot == header.txRoot:
# vmState.dumpDebuggingMetaData(header, body)
# warn "Validation error. Debugging metadata dumped."
let blockHash = header.blockHash() let blockHash = header.blockHash()
if NoPersistHeader notin flags: if NoPersistHeader notin flags:
if not c.db.persistHeader( if not c.db.persistHeader(

View File

@ -1,57 +0,0 @@
# Nimbus
# Copyright (c) 2019-2023 Status Research & Development GmbH
# 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.
import os, osproc, json
when defined(windows):
const
premixExecutable = "premix.exe"
browserLauncher = "cmd /c start"
elif defined(macos):
const
premixExecutable = "premix"
browserLauncher = "open"
else:
const
premixExecutable = "premix"
browserLauncher = "xdg-open"
proc getFileDir*(file: string): string =
var searchDirs = [
"." ,
"." / "build" ,
"." / "premix"
]
for dir in searchDirs:
if fileExists(dir / file):
return dir
result = ""
proc getFilePath(file: string): string =
let dir = getFileDir(file)
if dir.len > 0:
return dir / file
else:
return ""
proc launchPremix*(fileName: string, metaData: JsonNode) =
let premixExe = getFilePath(premixExecutable)
writeFile(fileName, metaData.pretty)
if premixExe.len > 0:
if execCmd(premixExe & " " & fileName) == 0:
if execCmd(browserLauncher & " " & getFilePath("index.html")) != 0:
echo "failed to launch default browser"
else:
echo "failed to execute the premix debugging tool"

View File

@ -23,7 +23,6 @@ import
./db/[core_db, ledger], ./db/[core_db, ledger],
./evm/[code_bytes, state, types], ./evm/[code_bytes, state, types],
./evm/tracer/legacy_tracer, ./evm/tracer/legacy_tracer,
./launcher,
./transaction, ./transaction,
./utils/utils ./utils/utils
@ -352,44 +351,6 @@ proc traceTransactionsImpl(
result.add traceTransactionImpl( result.add traceTransactionImpl(
com, header, transactions, i.uint64, {DisableState}) com, header, transactions, i.uint64, {DisableState})
proc dumpDebuggingMetaDataImpl(
vmState: BaseVMState;
blk: EthBlock;
launchDebugger = true;
) {.raises: [CatchableError].} =
template header: Header = blk.header
let
cc = activate CaptCtxRef.init(vmState.com, header)
blockNumber = header.number
bloom = createBloom(vmState.receipts)
defer: cc.release()
let blockSummary = %{
"receiptsRoot": %("0x" & toHex(calcReceiptsRoot(vmState.receipts).data)),
"stateRoot": %("0x" & toHex(vmState.stateDB.rootHash.data)),
"logsBloom": %("0x" & toHex(bloom))
}
var metaData = %{
"blockNumber": %blockNumber.toHex,
"txTraces": traceTransactionsImpl(vmState.com, header, blk.transactions),
"stateDump": dumpBlockStateImpl(vmState.com, blk),
"blockTrace": traceBlockImpl(vmState.com, blk, {DisableState}),
"receipts": toJson(vmState.receipts),
"block": blockSummary
}
metaData.dumpMemoryDB(cc.cpt)
let jsonFileName = "debug" & $blockNumber & ".json"
if launchDebugger:
launchPremix(jsonFileName, metaData)
else:
writeFile(jsonFileName, metaData.pretty())
# ------------------------------------------------------------------------------ # ------------------------------------------------------------------------------
# Public functions # Public functions
# ------------------------------------------------------------------------------ # ------------------------------------------------------------------------------
@ -443,14 +404,6 @@ proc traceTransactions*(
"traceTransactions".safeTracer: "traceTransactions".safeTracer:
result = com.traceTransactionsImpl(header, transactions) result = com.traceTransactionsImpl(header, transactions)
proc dumpDebuggingMetaData*(
vmState: BaseVMState;
blk: EthBlock;
launchDebugger = true;
) =
"dumpDebuggingMetaData".safeTracer:
vmState.dumpDebuggingMetaDataImpl(blk, launchDebugger)
# ------------------------------------------------------------------------------ # ------------------------------------------------------------------------------
# End # End
# ------------------------------------------------------------------------------ # ------------------------------------------------------------------------------

View File

@ -105,6 +105,12 @@ proc short*(h: Hash256): string =
bytes[^3..^1] = h.data[^3..^1] bytes[^3..^1] = h.data[^3..^1]
bytes.toHex bytes.toHex
proc short*(h: Bytes32): string =
var bytes: array[6, byte]
bytes[0..2] = h.data[0..2]
bytes[^3..^1] = h.data[^3..^1]
bytes.toHex
func short*(x: Duration): string = func short*(x: Duration): string =
let parts = x.toParts let parts = x.toParts
if parts[Hours] > 0: if parts[Hours] > 0:

8
premix/.gitignore vendored
View File

@ -1,8 +0,0 @@
*.db
*.db-lock
/output
/temp
/data
/nimcache
*.json
premixData.js

File diff suppressed because one or more lines are too long

Binary file not shown.

Before

Width:  |  Height:  |  Size: 189 KiB

View File

@ -1,520 +0,0 @@
var premix = function() {
function chunkSubstr(str, size) {
const numChunks = Math.ceil(str.length / size)
const chunks = new Array(numChunks)
for (let i = 0, o = 0; i < numChunks; ++i, o += size) {
chunks[i] = str.substr(o, size)
}
return chunks
}
function split32(text) {
if(text.length > 32) {
let chunks = chunkSubstr(text, 32);
let result = "";
for(var x of chunks) {
result += '<div>'+x+'</div>';
}
return result;
} else {
return text;
}
}
function renderCells(row, cls, cells) {
for(var text of cells) {
let cell = $(`<td ${cls}>${split32(text)}</td>`).appendTo(row);
if(text.length > 32) cell.addClass('tm-monospace-cell');
}
}
return {
fields: ['op', 'pc', 'gas', 'gasCost', 'depth'],
newTable: function(container) {
let table = $('<table class="uk-table uk-table-divider"/>').appendTo(container);
$('<thead><tr><th>Field</th><th>Nimbus</th><th>Geth</th></tr></thead>').appendTo(table);
return $('<tbody></tbody>').appendTo(table);
},
renderRow: function(body, nimbus, geth, x) {
let row = $('<tr/>').appendTo(body);
let ncr = nimbus instanceof Object ? nimbus[x].toString().toLowerCase() : nimbus;
let gcr = geth instanceof Object ? geth[x].toString().toLowerCase() : geth;
let cls = ncr == gcr ? '' : 'class="uk-text-danger"';
renderCells(row, cls, [x, ncr, gcr]);
},
newSection: function(container, title, colored) {
let section = $('<div class="uk-section uk-section-xsmall tm-horizontal-overflow"></div>').appendTo(container);
section.addClass(colored ? "uk-section-secondary uk-light" : "uk-section-muted");
let contentDiv = $('<div class="uk-container uk-margin-small-left uk-margin-small-right"></div>').appendTo(section);
$(`<h4>${title}</h4>`).appendTo(contentDiv);
return contentDiv;
}
};
}();
function deepCopy(src) {
return JSON.parse(JSON.stringify(src));
}
function windowResize() {
let bodyHeight = $(window).height();
$('#opCodeSideBar').css('height', parseInt(bodyHeight) - 80);
}
function renderTrace(title, nimbus, geth) {
let container = $('#opCodeContainer').empty();
let body = premix.newTable(container);
for(var x of premix.fields) {
premix.renderRow(body, nimbus, geth, x);
}
if(nimbus.error) {
geth.error = '';
premix.renderRow(body, nimbus, geth, 'error');
}
function renderExtra(name) {
let nk = Object.keys(nimbus[name]);
let gk = Object.keys(geth[name]);
let keys = new Set(nk.concat(gk));
if(keys.size > 0) {
let section = premix.newSection(container, name);
let body = premix.newTable(section);
for(var key of keys) {
premix.renderRow(body, nimbus[name], geth[name], key);
}
$('<hr class="uk-divider-icon">').appendTo(container);
}
}
renderExtra("memory");
renderExtra("stack");
renderExtra("storage");
}
function opCodeRenderer(txId, nimbus, geth) {
function analyzeList(nimbus, geth) {
for(var i in nimbus) {
if(nimbus[i].toString().toLowerCase() != geth[i].toString().toLowerCase()) return false;
}
return true;
}
function fillEmptyList(a, b) {
if(a.length > b.length) {
for(var i in a) {
if(b[i] === undefined) {
b[i] = '';
}
}
}
}
function fillEmptyMap(a, b) {
if(Object.keys(a).length > Object.keys(b).length) {
for(var i in a) {
if(b[i] === undefined) {
b[i] = '';
}
}
}
}
function fillEmptyField(nimbus, geth) {
if(nimbus.memory === undefined) {
nimbus.memory = [];
}
if(geth.memory === undefined) {
geth.memory = [];
}
if(nimbus.stack === undefined) {
nimbus.stack = [];
}
if(geth.stack === undefined) {
geth.stack = [];
}
if(nimbus.storage === undefined) {
nimbus.storage = {};
}
if(geth.storage === undefined) {
geth.storage = {};
}
fillEmptyList(nimbus.memory, geth.memory);
fillEmptyList(geth.memory, nimbus.memory);
fillEmptyList(nimbus.stack, geth.stack);
fillEmptyList(geth.stack, nimbus.stack);
fillEmptyMap(nimbus.storage, geth.storage);
fillEmptyMap(geth.storage, nimbus.storage);
}
function moveStack(ncs, gcs, i) {
let idx = parseInt(i);
ncs[idx-1].stack = deepCopy(ncs[idx].stack);
gcs[idx-1].stack = deepCopy(gcs[idx].stack);
}
function analyze(nimbus, geth) {
for(var x of premix.fields) {
if(nimbus[x] === undefined) nimbus[x] = '';
if(geth[x] === undefined) geth[x] = '';
if(nimbus[x].toString().toLowerCase() != geth[x].toString().toLowerCase()) return false;
}
let result = analyzeList(nimbus.memory, geth.memory);
result = result && analyzeList(nimbus.stack, geth.stack);
result = result && analyzeList(nimbus.storage, geth.storage);
return result;
}
txId = parseInt(txId);
$('#opCodeTitle').text(`Tx #${(txId+1)}`);
const numRows = Math.max(nimbus.txTraces[txId].structLogs.length, geth.txTraces[txId].structLogs.length);
if(numRows == 0) {
$('#opCodeContainer').empty();
$('#paging').empty();
$('#opCodeSideBar').empty();
return;
}
const rowsPerPage = 500;
var numPages = numRows / rowsPerPage;
if(numRows % rowsPerPage != 0) numPages++;
$("#paging").paging(numRows, {
format: numPages < 10 ? "n".repeat(numPages) : '[< (qq -) nnncnnn (- pp) >]',
perpage: rowsPerPage,
lapping: 1,
page: 1,
onSelect: function (page) {
const data = this.slice;
const start = data[0];
const stop = data[1];
var ncs = deepCopy(nimbus.txTraces[txId].structLogs.slice(start, stop));
var gcs = deepCopy(geth.txTraces[txId].structLogs.slice(start, stop));
var sideBar = $('#opCodeSideBar').empty();
function fillEmptyOp(a, b) {
function emptyOp() {
return {op: '', pc: '', gas: '', gasCost: '', depth: '',
storage:{}, memory: [], stack: []};
}
if(a.length > b.length) {
for(var i in a) {
if(b[i] === undefined) {
b[i] = emptyOp();
}
}
}
}
fillEmptyOp(ncs, gcs);
fillEmptyOp(gcs, ncs);
for(var i in ncs) {
fillEmptyField(ncs[i], gcs[i]);
if(parseInt(i) > 0) {
moveStack(ncs, gcs, i);
}
}
for(var i in ncs) {
let pc = ncs[i].pc == '' ? gcs[i].pc : ncs[i].pc;
let op = ncs[i].op == '' ? gcs[i].op : ncs[i].op;
if(!analyze(ncs[i], gcs[i])) {
var nav = $(`<li><a class="tm-text-danger" rel="${i}" href="#">${pc + ' ' + op}</a></li>`).appendTo(sideBar);
} else {
var nav = $(`<li><a rel="${i}" href="#">${pc + ' ' + op}</a></li>`).appendTo(sideBar);
}
nav.children('a').click(function(ev) {
let idx = this.rel;
$('#opCodeSideBar li').removeClass('uk-active');
$(this).parent().addClass('uk-active');
renderTrace('tx', ncs[idx], gcs[idx]);
});
}
if(ncs.length > 0) {
renderTrace("tx", ncs[0], gcs[0]);
} else {
$('#opCodeContainer').empty();
}
},
onFormat: function (type) {
switch (type) {
case 'block': // n and c
if (this.value == this.page) {
return '<li class="uk-active"><span>' + this.value + '</span></li>';
} else {
return '<li><a href="#">' + this.value + '</a></li>';
}
case 'next': // >
return '<li><a href="#"><span uk-pagination-next></span></a></li>';
case 'prev': // <
return '<li><a href="#"><span uk-pagination-previous></span></a></li>';
case 'first': // [
return '<li><a href="#">first</a></li>';
case 'last': // ]
return '<li><a href="#">last</a></li>';
case "leap":
return " ";
case 'fill':
return '<li class="uk-disabled"><span>...</span></li>';
case 'left':
if(this.value >= this.page) return '';
return '<li><a href="#">' + this.value + '</a></li>';
case 'right':
if(this.value <= this.page) return '';
return '<li><a href="#">' + this.value + '</a></li>';
}
}
});
windowResize();
}
function transactionsRenderer(txId, nimbus, geth) {
txId = parseInt(txId);
$('#transactionsTitle').text(`Tx #${(txId+1)}`);
let container = $('#transactionsContainer').empty();
function renderTx(nimbus, geth) {
let body = premix.newTable(container);
const fields = ["gas", "returnValue", "cumulativeGasUsed", "bloom"];
for(var x of fields) {
premix.renderRow(body, nimbus, geth, x);
}
$('<hr class="uk-divider-icon">').appendTo(container);
if(nimbus.root || geth.root) {
if(geth.root === undefined) geth.root = '';
if(nimbus.root == undefined) nimbus.root = '';
premix.renderRow(body, nimbus, geth, 'root');
}
if(nimbus.status || geth.status) {
if(geth.status === undefined) geth.status = '';
if(nimbus.status == undefined) nimbus.status = '';
premix.renderRow(body, nimbus, geth, 'status');
}
function fillEmptyLogs(a, b) {
function emptyLog() {
return {address: '', topics: [], data: ''};
}
if(a.logs.length > b.logs.length) {
for(var i in a.logs) {
if(b.logs[i] === undefined) {
b.logs[i] = emptyLog();
}
}
}
}
fillEmptyLogs(geth, nimbus);
fillEmptyLogs(nimbus, geth);
for(var i in nimbus.logs) {
$(`<h4>Receipt Log #${i}</h4>`).appendTo(container);
let a = nimbus.logs[i];
let b = geth.logs[i];
//console.log(a.topics);
a.topics = a.topics.join(',');
b.topics = b.topics.join(',');
let body = premix.newTable(container);
premix.renderRow(body, a, b, 'address');
premix.renderRow(body, a, b, 'data');
premix.renderRow(body, a, b, 'topics');
$('<hr class="uk-divider-icon">').appendTo(container);
}
}
let tx = geth.block.transactions[txId];
let ntx = nimbus.txTraces[txId];
let gtx = geth.txTraces[txId];
if(ntx.returnValue.length == 0) {
ntx.returnValue = "0x";
}
let ncr = $.extend({
gas: ntx.gas,
returnValue: ntx.returnValue
},
deepCopy(nimbus.receipts[txId])
);
let gcr = $.extend({
gas: gtx.gas,
returnValue: "0x" + gtx.returnValue
},
deepCopy(geth.receipts[txId])
);
$(`<h4>Transaction Kind: ${tx.txKind}</h4>`).appendTo(container);
renderTx(ncr, gcr);
}
function accountsRenderer(nimbus, geth) {
function emptyAccount() {
return {
address: '',
nonce: '',
balance: '',
codeHash: '',
code: '',
storageRoot: '',
storage: {}
};
}
function precompiledContractsName(address) {
switch(address) {
case "0x0000000000000000000000000000000000000001": return "ecRecover";
case "0x0000000000000000000000000000000000000002": return "Sha256";
case "0x0000000000000000000000000000000000000003": return "RipeMd160";
case "0x0000000000000000000000000000000000000004": return "Identity";
case "0x0000000000000000000000000000000000000005": return "ModExp";
case "0x0000000000000000000000000000000000000006": return "bn256ecAdd";
case "0x0000000000000000000000000000000000000007": return "bn256ecMul";
case "0x0000000000000000000000000000000000000008": return "bn256ecPairing";
default: return "";
}
}
let container = $('#accountsContainer').empty();
$('#accountsTitle').text('Block #' + parseInt(geth.block.number, 16));
let ncs = deepCopy(nimbus.stateDump.after);
let gcs = deepCopy(geth.accounts);
let accounts = [];
for(var address in ncs) {
let n = ncs[address];
n.address = address;
if(gcs[address]) {
let geth = gcs[address];
geth.address = address;
accounts.push({name: n.name, nimbus: n, geth: geth});
delete gcs[address];
} else {
accounts.push({name: n.name, nimbus: n, geth: emptyAccount()});
}
}
for(var address in gcs) {
let geth = gcs[address];
geth.address = address;
accounts.push({name: "unknown", nimbus: emptyAccount(), geth: geth});
}
for(var acc of accounts) {
let pa = precompiledContractsName(acc.nimbus.address);
let precompiledContract = pa == '' ? '' : ` or Precompiled Contract(${pa})`;
$(`<h4>Account Name: ${acc.name}${precompiledContract}</h4>`).appendTo(container);
let body = premix.newTable(container);
const fields = ['address', 'nonce', 'balance', 'codeHash', 'code', 'storageRoot'];
for(var x of fields) {
premix.renderRow(body, acc.nimbus, acc.geth, x);
}
let storage = [];
let nss = acc.nimbus.storage;
let gss = acc.geth.storage;
for(var idx in nss) {
if(gss[idx]) {
storage.push({idx: idx, nimbus: nss[idx], geth: gss[idx]});
delete gss[idx];
} else {
if(nss[idx] != "0x0000000000000000000000000000000000000000000000000000000000000000") {
storage.push({idx: idx, nimbus: nss[idx], geth: ''});
}
}
}
for(var idx in gss) {
if(gss[idx] != "0x0000000000000000000000000000000000000000000000000000000000000000") {
storage.push({idx: idx, nimbus: '', geth: gss[idx]});
}
}
if(storage.length > 0) {
$(`<h4>${acc.name} Storage</h4>`).appendTo(container);
let body = premix.newTable(container);
for(var s of storage) {
premix.renderRow(body, s.nimbus, s.geth, s.idx);
}
}
$('<hr class="uk-divider-icon">').appendTo(container);
}
}
function headerRenderer(nimbus, geth) {
let container = $('#headerContainer').empty();
$('#headerTitle').text('Block #' + parseInt(geth.block.number, 16));
let body = premix.newTable(container);
const blockSummary = ['stateRoot', 'receiptsRoot', 'logsBloom'];
for(var idx of blockSummary) {
premix.renderRow(body, nimbus.block, geth.block, idx);
}
}
function generateNavigation(txs, nimbus, geth) {
function navAux(menuId, renderer) {
let menu = $(menuId).click(function(ev) {
renderer(0, nimbus, geth);
});
if(txs.length == 0) {
menu.parent().addClass('uk-disabled');
} else if(txs.length > 1) {
$('<span uk-icon="icon: triangle-down"></span>').appendTo(menu);
let div = $('<div uk-dropdown="mode: hover;"/>').appendTo(menu.parent());
let list = $('<ul class="uk-nav uk-dropdown-nav"/>').appendTo(div);
for(var i in txs) {
let id = parseInt(i) + 1;
$(`<li class="uk-active"><a rel="${i}" href="#">TX #${id}</a></li>`).appendTo(list);
}
list.find('li a').click(function(ev) {
renderer(this.rel, nimbus, geth);
});
}
}
navAux('#opCodeMenu', opCodeRenderer);
navAux('#transactionsMenu', transactionsRenderer);
$('#accountsMenu').click(function(ev) {
accountsRenderer(nimbus, geth);
});
$('#headerMenu').click(function(ev) {
headerRenderer(nimbus, geth);
});
}
$(document).ready(function() {
var nimbus = premixData.nimbus;
var geth = premixData.geth;
var transactions = geth.block.transactions;
generateNavigation(transactions, nimbus, geth);
});

File diff suppressed because one or more lines are too long

View File

@ -1,14 +0,0 @@
/*
jQuery paging plugin v1.3.0 23/06/2014
http://www.xarg.org/2011/09/jquery-pagination-revised/
Copyright (c) 2011, Robert Eisele (robert@xarg.org)
Dual licensed under the MIT or GPL Version 2 licenses.
*/
(function(n,v,r){n.fn.paging=function(z,A){var t=this,b={setOptions:function(a){b.a=n.extend(b.a||{lapping:0,perpage:10,page:1,refresh:{interval:10,url:null},format:"",lock:!1,circular:!1,onClick:null,onFormat:function(){},onSelect:function(){return!0},onRefresh:function(){}},a||{});b.a.lapping|=0;b.a.perpage|=0;null!==b.a.page&&(b.a.page|=0);1>b.a.perpage&&(b.a.perpage=10);b.interval&&v.clearInterval(b.interval);b.a.refresh.url&&(b.interval=v.setInterval(function(){n.ajax({url:b.a.refresh.url,success:function(a){if("string"===
typeof a)try{a=n.parseJSON(a)}catch(m){return}b.a.onRefresh(a)}})},1E3*b.a.refresh.interval));b.format=function(a){for(var b=0,f=0,h=1,g={g:[],i:0,h:0,b:5,current:3,l:0,m:0},c,p=/[*<>pq\[\]().-]|[nc]+!?/g,n={"[":"first","]":"last","<":"prev",">":"next",q:"left",p:"right","-":"fill",".":"leap"},e={};c=p.exec(a);)c=""+c,r===n[c]?"("===c?f=++b:")"===c?f=0:h&&("*"===c?(g.i=1,g.h=0):(g.i=0,g.h="!"===c.charAt(c.length-1),g.b=c.length-g.h,(g.current=1+c.indexOf("c"))||(g.current=1+g.b>>1)),g.g.push({f:"block",
j:0,c:0}),h=0):(g.g.push({f:n[c],j:f,c:r===e[c]?e[c]=1:++e[c]}),"q"===c?++g.m:"p"===c&&++g.l);return g}(b.a.format);return b},setNumber:function(a){b.s=r===a||0>a?-1:a;return b},setPage:function(a){function w(a,b,c){c=""+a.onFormat.call(b,c);p=b.value?p+c.replace(/<a/i,'<a data-page="'+b.value+'"'):p+c}if(b.a.lock)return b.a.onSelect(0,t),b;if(r===a){if(a=b.a.page,null===a)return b}else if(b.a.page==a)return b;b.a.page=a|=0;var m=b.s,f=b.a,h,g,c,p,x=1,e=b.format,d,k,l,q,y=e.g.length,u=y;f.perpage<=
f.lapping&&(f.lapping=f.perpage-1);q=m<=f.lapping?0:f.lapping|0;0>m?(c=m=-1,h=Math.max(1,a-e.current+1-q),g=h+e.b):(c=1+Math.ceil((m-f.perpage)/(f.perpage-q)),a=Math.max(1,Math.min(0>a?1+c+a:a,c)),e.i?(h=1,g=1+c,e.current=a,e.b=c):(h=Math.max(1,Math.min(a-e.current,c-e.b)+1),g=e.h?h+e.b:Math.min(h+e.b,1+c)));for(;u--;){k=0;l=e.g[u];switch(l.f){case "left":k=l.c<h;break;case "right":k=g<=c-e.l+l.c;break;case "first":k=e.current<a;break;case "last":k=e.b<e.current+c-a;break;case "prev":k=1<a;break;
case "next":k=a<c}x|=k<<l.j}d={number:m,lapping:q,pages:c,perpage:f.perpage,page:a,slice:[(k=a*(f.perpage-q)+q)-f.perpage,Math.min(k,m)]};for(p="";++u<y;){l=e.g[u];k=x>>l.j&1;switch(l.f){case "block":for(;h<g;++h)d.value=h,d.pos=1+e.b-g+h,d.active=h<=c||0>m,d.first=1===h,d.last=h===c&&0<m,w(f,d,l.f);continue;case "left":d.value=l.c;d.active=l.c<h;break;case "right":d.value=c-e.l+l.c;d.active=g<=d.value;break;case "first":d.value=1;d.active=k&&1<a;break;case "prev":(d.active=f.circular)?d.value=1===
a?c:a-1:(d.value=Math.max(1,a-1),d.active=k&&1<a);break;case "last":(d.active=0>m)?d.value=1+a:(d.value=c,d.active=k&&a<c);break;case "next":(d.active=f.circular)?d.value=1+a%c:(d.active=0>m)?d.value=1+a:(d.value=Math.min(1+a,c),d.active=k&&a<c);break;case "leap":case "fill":d.pos=l.c;d.active=k;w(f,d,l.f);continue}d.pos=l.c;d.last=d.first=r;w(f,d,l.f)}t.length&&(n("a",t.html(p)).click(f.onClick||function(a){a.preventDefault();a=this;do if("a"===a.nodeName.toLowerCase())break;while(a=a.parentNode);
b.setPage(n(a).data("page"));b.o&&(v.location=a.href)}),b.o=f.onSelect.call({number:m,lapping:q,pages:c,slice:d.slice},a,t));return b}};return b.setNumber(z).setOptions(A).setPage()}})(jQuery,this);

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

View File

@ -1,134 +0,0 @@
# Nimbus
# Copyright (c) 2020-2024 Status Research & Development GmbH
# 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.
import
std/[os, parseopt, strutils],
eth/common,
stint,
chronicles,
../nimbus/config
from ../nimbus/common/chain_config import
MainNet,
SepoliaNet,
HoleskyNet
type
ConfigStatus* = enum
## Configuration status flags
Success, ## Success
EmptyOption, ## No options in category
ErrorUnknownOption, ## Unknown option in command line found
ErrorParseOption, ## Error in parsing command line option
ErrorIncorrectOption, ## Option has incorrect value
Error ## Unspecified error
PremixConfiguration* = ref object
dataDir*: string
head*: BlockNumber
maxBlocks*: int
numCommits*: int
netId*: NetworkId
var premixConfig {.threadvar.}: PremixConfiguration
proc getConfiguration*(): PremixConfiguration {.gcsafe.}
proc processInteger(v: string, o: var int): ConfigStatus =
## Convert string to integer.
try:
o = parseInt(v)
result = Success
except ValueError:
result = ErrorParseOption
proc initConfiguration(): PremixConfiguration =
result = new PremixConfiguration
const dataDir = defaultDataDir()
result.dataDir = dataDir
result.head = 0'u64
result.maxBlocks = 0
result.numCommits = 128
result.netId = MainNet
proc getConfiguration*(): PremixConfiguration =
if isNil(premixConfig):
premixConfig = initConfiguration()
result = premixConfig
proc processBlockNumber(val: string, o: var BlockNumber): ConfigStatus =
if val.len > 2 and val[0] == '0' and val[1] == 'x':
o = UInt256.fromHex(val).truncate(BlockNumber)
else:
o = parse(val, UInt256).truncate(BlockNumber)
result = Success
func processNetId(val: string, o: var NetworkId): ConfigStatus =
case val.toLowerAscii()
of "main": o = MainNet
of "sepolia": o = SepoliaNet
of "holesky": o = HoleskyNet
template checkArgument(fun, o, value: untyped) =
## Checks if arguments got processed successfully
let res = fun(value, o)
if res == Success:
continue
elif res == ErrorParseOption:
msg = "Error processing option [" & key & "] with value [" & value & "]"
result = res
break
elif res == ErrorIncorrectOption:
msg = "Incorrect value for option [" & key & "] value [" & value & "]"
result = res
break
proc processArguments*(msg: var string): ConfigStatus =
var
opt = initOptParser()
length = 0
config = getConfiguration()
result = Success
for kind, key, value in opt.getopt():
case kind
of cmdArgument:
return EmptyOption
of cmdLongOption, cmdShortOption:
inc(length)
case key.toLowerAscii()
of "help":
return EmptyOption
of "datadir": config.dataDir = value
of "maxblocks":
checkArgument(processInteger, config.maxBlocks, value)
of "head":
checkArgument(processBlockNumber, config.head, value)
of "numcommits":
checkArgument(processInteger, config.numCommits, value)
config.numCommits = max(config.numCommits, 512)
of "netid":
checkArgument(processNetId, config.netId, value)
else:
msg = "Unknown option " & key
if value.len > 0: msg = msg & " : " & value
return ErrorUnknownOption
of cmdEnd:
msg = "Error processing option [" & key & "]"
return ErrorParseOption
info "Using configuration parameters: ",
datadir = config.dataDir,
maxblocks = config.maxBlocks,
head = config.head,
numcommits = config.numCommits,
netid = config.netId

View File

@ -1,79 +0,0 @@
# Nimbus
# Copyright (c) 2020-2024 Status Research & Development GmbH
# 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.
import
std/[json, os, strutils],
stew/byteutils,
chronicles,
results,
../nimbus/[evm/state, evm/types],
../nimbus/core/executor,
./premixcore, ./prestate,
../nimbus/tracer,
../nimbus/common/common
proc prepareBlockEnv(node: JsonNode, memoryDB: CoreDbRef) =
let state = node["state"]
let kvt = memoryDB.ctx.getKvt()
for k, v in state:
let key = hexToSeqByte(k)
let value = hexToSeqByte(v.getStr())
kvt.put(key, value).isOkOr:
raiseAssert "prepareBlockEnv(): put() (loop) failed " & $$error
proc executeBlock(blockEnv: JsonNode, memoryDB: CoreDbRef, blockNumber: BlockNumber) =
var
parentNumber = blockNumber - 1
com = CommonRef.new(memoryDB)
parent = com.db.getBlockHeader(parentNumber)
blk = com.db.getEthBlock(blockNumber)
let transaction = memoryDB.ctx.newTransaction()
defer: transaction.dispose()
let
vmState = BaseVMState.new(parent, blk.header, com)
validationResult = vmState.processBlock(blk)
if validationResult.isErr:
error "block validation error", err = validationResult.error()
else:
info "block validation success", blockNumber
transaction.rollback()
vmState.dumpDebuggingMetaData(blk, false)
let
fileName = "debug" & $blockNumber & ".json"
nimbus = json.parseFile(fileName)
geth = blockEnv["geth"]
processNimbusData(nimbus)
# premix data goes to report page
generatePremixData(nimbus, geth)
# prestate data goes to debug tool and contains data
# needed to execute single block
generatePrestate(nimbus, geth, blockNumber, parent, blk)
proc main() =
if paramCount() == 0:
echo "usage: debug blockxxx.json"
quit(QuitFailure)
let
blockEnv = json.parseFile(paramStr(1))
memoryDB = newCoreDbRef(DefaultDbMemory)
blockNumberHex = blockEnv["blockNumber"].getStr()
blockNumber = parseHexInt(blockNumberHex).uint64
prepareBlockEnv(blockEnv, memoryDB)
executeBlock(blockEnv, memoryDB, blockNumber)
main()

View File

@ -1,70 +0,0 @@
# Nimbus
# Copyright (c) 2020-2024 Status Research & Development GmbH
# 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.
#
# helper tool to dump debugging data for persisted block
# usage: dumper [--datadir:your_path] --head:blockNumber
#
import
results,
../nimbus/common/common,
../nimbus/db/opts,
../nimbus/db/core_db/persistent,
../nimbus/core/executor,
../nimbus/[evm/state, evm/types],
../nimbus/tracer,
./configuration # must be late (compilation annoyance)
proc dumpDebug(com: CommonRef, blockNumber: BlockNumber) =
var
capture = com.db.newCapture.value
captureCom = com.clone(capture.recorder)
let transaction = capture.recorder.ctx.newTransaction()
defer: transaction.dispose()
var
parentNumber = blockNumber - 1
parent = captureCom.db.getBlockHeader(parentNumber)
blk = captureCom.db.getEthBlock(blockNumber)
vmState = BaseVMState.new(parent, blk.header, captureCom)
discard captureCom.db.setHead(parent, true)
discard vmState.processBlock(blk)
transaction.rollback()
vmState.dumpDebuggingMetaData(blk, false)
proc main() {.used.} =
let conf = getConfiguration()
let com = CommonRef.new(
newCoreDbRef(DefaultDbPersistent, conf.dataDir, DbOptions.init()))
if conf.head != 0'u64:
dumpDebug(com, conf.head)
when isMainModule:
var message: string
## Processing command line arguments
if processArguments(message) != Success:
echo message
quit(QuitFailure)
else:
if len(message) > 0:
echo message
quit(QuitSuccess)
try:
main()
except:
echo getCurrentExceptionMsg()

View File

@ -1,118 +0,0 @@
# Nimbus
# Copyright (c) 2021-2024 Status Research & Development GmbH
# 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.
import
std/json,
chronos, results, eth/common,
graphql/httpclient,
./parser
const ethQuery = """
fragment headerFields on Block {
parentHash: parent { value: hash }
sha3Uncles: ommerHash
miner { value: address }
stateRoot
transactionsRoot
receiptsRoot
logsBloom
difficulty
number
gasLimit
gasUsed
timestamp
extraData
mixHash
nonce
baseFeePerGas # EIP-1559
}
query getBlock($blockNumber: Long!) {
chainID # EIP-1559
block(number: $blockNumber) {
... headerFields
ommerCount
ommers {
... headerFields
}
transactionCount
transactions {
nonce
gasPrice
gas
to {value: address}
value
input: inputData
v
r
s
maxFeePerGas # EIP-1559
maxPriorityFeePerGas # EIP-1559
effectiveGasPrice # EIP-1559
type
hash
from {value: address}
accessList {
address
storageKeys
}
index
}
}
}
"""
type
Block* = object
header*: BlockHeader
body*: BlockBody
proc fromJson(_: type ChainId, n: JsonNode, name: string): ChainId =
var chainId: int
fromJson(n, name, chainId)
ChainId(chainId)
proc requestBlock*(blockNumber: BlockNumber, parseTx = true): Block =
let address = initTAddress("127.0.0.1:8545")
let clientRes = GraphqlHttpClientRef.new(address)
if clientRes.isErr:
raise newException(ValueError, clientRes.error)
let client = clientRes.get()
client.addVar("blockNumber", $blockNumber)
let res = waitFor client.sendRequest(ethQuery)
if res.isErr:
raise newException(ValueError, res.error)
let resp = res.get()
let n = json.parseJson(resp.response)
if n.hasKey("errors"):
debugEcho n.pretty
quit(1)
let nh = n["data"]["block"]
let chainId = ChainId.fromJson(n["data"], "chainID")
result.header = parseBlockHeader(nh)
if parseTx:
let txs = nh["transactions"]
for txn in txs:
var tx = parseTransaction(txn)
tx.chainId = chainId
validateTxSenderAndHash(txn, tx)
result.body.transactions.add tx
let uncles = nh["ommers"]
for un in uncles:
result.body.uncles.add parseBlockHeader(un)
waitFor client.closeWait()

View File

@ -1,15 +0,0 @@
# Nimbus
# Copyright (c) 2024 Status Research & Development GmbH
# 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.
# Currently disabled (moved to no-hunter.nim)
# Would never haved worked on Aristo. Needs to maintain the account with
# the state db, i.e. using `CoreDbAccount` for `Account` (see
# `state_db/base.nim`.)

View File

@ -1,181 +0,0 @@
<html class="uk-height-1-1">
<head>
<title id="windowTitle">Premix Report Page</title>
<meta http-equiv="Content-type" content="text/html; charset=utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<script src="assets/js/jquery.min.js"></script>
<script src="assets/js/jquery.paging.min.js"></script>
<script src="assets/js/uikit.min.js"></script>
<script src="assets/js/uikit-icons.min.js"></script>
<script src="assets/js/index.js"></script>
<script src="premixData.js"></script>
<link rel="stylesheet" href="assets/css/uikit.min.css" />
<style>
body {
font: 12px normal Arial, Helvetica, sans-serif;
}
.tm-horizontal-overflow {
overflow-x: auto;
}
.tm-sidebar {
position: fixed;
overflow-y: auto;
padding-left:40px;
}
.tm-monospace-cell {
font-family: "Courier New", Courier, monospace;
font-weight: 500;
}
.uk-nav-default > li > a.tm-text-danger {
color: #f0506e;
}
.uk-nav-default > li > a.tm-text-danger:hover {
color: red;
}
.uk-nav-default > li.uk-active > a.tm-text-danger {
color: red;
}
</style>
</head>
<body onresize="windowResize()" class="uk-height-1-1">
<div class="uk-section-small uk-background-primary uk-light" uk-sticky="bottom: #offset">
<!-- Navigation -->
<div class="uk-overlay uk-position-left uk-flex uk-flex-middle">
<h1>Premix Report Page</h1>
</div>
<div class="uk-position-right uk-overlay">
<ul class="uk-subnav uk-subnav-divider" uk-switcher="connect:#switcherSection">
<li><a id="opCodeMenu" href="#">OpCode</a></li>
<li><a id="transactionsMenu" href="#">Transactions</a></li>
<li><a id="accountsMenu" href="#">Accounts</a></li>
<li><a id="headerMenu" href="#">Header</a></li>
<li class="uk-active"><a href="#">Help</a></li>
</ul>
</div>
<!-- Navigation -->
</div>
<ul id="switcherSection" class="uk-switcher">
<li>
<!-- Opcode Page -->
<div class="uk-grid-collapse" uk-grid>
<div class="uk-width-1-5@m">
<ul id="opCodeSideBar" class="tm-sidebar uk-nav uk-nav-default uk-height-1-1 uk-width-1-5@m">
<!-- op code traces navigation sidebar -->
</ul>
</div>
<div class="uk-width-4-5@m">
<div class="uk-section uk-section-small uk-section-secondary uk-light">
<ul id="paging" class="uk-pagination uk-flex-center" uk-margin>
</ul>
</div>
<div class="uk-section-small uk-section-default">
<div class="uk-container uk-container-expand">
<h3>Opcode Trace <span id="opCodeTitle" class="uk-text-primary uk-text-small">Tx #</span></h3>
<div id="opCodeContainer">
</div>
</div>
</div>
<div class="uk-section uk-section-small uk-section-secondary uk-light">
<div class="uk-container uk-text-center">
<h2>Have You Found The Bug?</h2>
</div>
<ul class="uk-subnav uk-subnav-divider uk-flex uk-flex-center" uk-margin>
<li><a href="https://github.com/status-im/nimbus"><span uk-icon="github" class="uk-margin-small-right"></span>Github</a></li>
<li><a href="https://gitter.im/status-im/nimbus"><span uk-icon="gitter" class="uk-margin-small-right"></span>Gitter</a></li>
</ul>
</div>
</div>
</div>
<!-- Opcode Page -->
</li>
<li>
<!-- Transactions Page -->
<div class="uk-section-small uk-section-default">
<div class="uk-container uk-container-medium">
<h3>Transaction's Receipts <span id="transactionsTitle" class="uk-text-primary uk-text-small">Tx #</span></h3>
<div id="transactionsContainer">
</div>
</div>
</div>
<!-- Transactions Page -->
</li>
<li>
<!-- Accounts Page -->
<div class="uk-section-small uk-section-default">
<div class="uk-container uk-container-medium">
<h3>Post State Accounts <span id="accountsTitle" class="uk-text-primary uk-text-small">Block #</span></h3>
<div id="accountsContainer">
</div>
</div>
</div>
<!-- Accounts Page -->
</li>
<li>
<!-- Header Page -->
<div class="uk-section-small uk-section-default">
<div class="uk-container uk-container-medium">
<h3>Header Summary <span id="headerTitle" class="uk-text-primary uk-text-small">Block #</span></h3>
<div id="headerContainer">
</div>
</div>
</div>
<!-- Header Page -->
</li>
<li>
<!-- Help Page -->
<div class="uk-section-small uk-section-default">
<div class="uk-container uk-container-xsmall">
<h2>Help</h2>
<p>
Work your way through the top-right menu, left to right, to find out where the bug might be located.
If you see <span class="uk-text-danger">red colored text</span>, it means you already found the difference between Nimbus and the other Ethereum
client's tracing result.
</p>
<p>
If there is no red colored text in the <strong>OPCODE</strong> section, it means the bug might be located in the <strong>TRANSACTIONS</strong> section, or in the <strong>HEADER</strong> section.
</p>
<p>
Once you locate the bug, you can use the <span class="uk-text-primary">./build/debug</span> tool
to sort things out until there are no more errors and the block passes validation.
</p>
<p>
Blocks with multiple transactions will have submenus in the navigation bar.
Usually, only the first transaction with red colored text is problematic, but it might affect the
other transactions. In the <strong>OPCODE</strong> section, the same thing happens. Perhaps only the first red-colored instruction
is problematic, but it will affect the other instructions.
</p>
<p>
Transactions in the <strong>TRANSACTIONS</strong> section are marked as: <span class="uk-text-warning">Regular, ContractCreation, or ContractCall</span>.
Each kind is processed separately by Nimbus, in different procedures.
</p>
</div>
</div>
<!-- Help Page -->
</li>
</ul>
</body>
</html>

View File

@ -1,78 +0,0 @@
# Nimbus
# Copyright (c) 2020-2023 Status Research & Development GmbH
# 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.
const postStateTracer* = """{
postState: {},
// lookupAccount injects the specified account into the postState object.
lookupAccount: function(addr, db){
var acc = toHex(addr);
if (this.postState[acc] === undefined) {
this.postState[acc] = {
code: toHex(db.getCode(addr)),
storage: {}
};
}
},
// lookupStorage injects the specified storage entry of the given account into
// the postState object.
lookupStorage: function(addr, key, db){
var acc = toHex(addr);
var idx = toHex(key);
this.lookupAccount(addr, db);
if (this.postState[acc].storage[idx] === undefined) {
// bug in geth js tracer
// we will use eth_getProof to fill the storage later
this.postState[acc].storage[idx] = "";
}
},
// result is invoked when all the opcodes have been iterated over and returns
// the final result of the tracing.
result: function(ctx, db) {
this.lookupAccount(ctx.from, db);
this.lookupAccount(ctx.to, db);
// Return the assembled allocations (postState)
return this.postState;
},
// step is invoked for every opcode that the VM executes.
step: function(log, db) {
// Add the current account if we just started tracing
if (this.postState === null){
this.postState = {};
// Balance will potentially be wrong here, since this will include the value
// sent along with the message. We fix that in 'result()'.
this.lookupAccount(log.contract.getAddress(), db);
}
// Whenever new state is accessed, add it to the postState
switch (log.op.toString()) {
case "EXTCODECOPY": case "EXTCODESIZE": case "BALANCE":
this.lookupAccount(toAddress(log.stack.peek(0).toString(16)), db);
break;
case "CREATE":
var from = log.contract.getAddress();
this.lookupAccount(toContract(from, db.getNonce(from)), db);
break;
case "CALL": case "CALLCODE": case "DELEGATECALL": case "STATICCALL":
this.lookupAccount(toAddress(log.stack.peek(1).toString(16)), db);
break;
case 'SSTORE':case 'SLOAD':
this.lookupStorage(log.contract.getAddress(), toWord(log.stack.peek(0).toString(16)), db);
break;
}
},
// fault is invoked when the actual execution of an opcode fails.
fault: function(log, db) {}
}
"""

View File

@ -1,164 +0,0 @@
# Nimbus
# Copyright (c) 2020-2024 Status Research & Development GmbH
# 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.
import
std/[json, tables, hashes],
eth/trie/trie_defs,
stint, stew/byteutils, chronicles,
../nimbus/[evm/state, evm/types],
../nimbus/utils/utils,
../nimbus/tracer,
../nimbus/db/[core_db, state_db/read_write],
../nimbus/core/executor,
../nimbus/common/common,
"."/[configuration, downloader, parser, premixcore]
const
emptyCodeHash = blankStringHash
proc store(memoryDB: CoreDbRef, branch: JsonNode) =
for p in branch:
let rlp = hexToSeqByte(p.getStr)
let hash = keccakHash(rlp)
memoryDB.kvt.put(hash.data, rlp)
proc parseAddress(address: string): EthAddress =
hexToByteArray(address, result)
proc parseU256(val: string): UInt256 =
UInt256.fromHex(val)
proc prepareBlockEnv(parent: BlockHeader, thisBlock: Block): CoreDbRef =
var
accounts = requestPostState(thisBlock)
memoryDB = newCoreDbRef DefaultDbMemory
accountDB = newAccountStateDB(memoryDB, parent.stateRoot)
parentNumber = %(parent.number.prefixHex)
for address, account in accounts:
updateAccount(address, account, parent.number)
let
accountProof = account["accountProof"]
storageProof = account["storageProof"]
address = parseAddress(address)
acc = parseAccount(account)
memoryDB.store(accountProof)
accountDB.setAccount(address, acc)
for storage in storageProof:
let
key = parseU256(storage["key"].getStr)
val = parseU256(storage["value"].getStr)
proof = storage["proof"]
memoryDB.store(proof)
accountDB.setStorage(address, key, val)
if acc.codeHash != emptyCodeHash:
let codeStr = request("eth_getCode", %[%address.prefixHex, parentNumber])
let code = hexToSeqByte(codeStr.getStr)
accountDB.setCode(address, code)
accountDB.setAccount(address, acc)
result = memoryDB
type
HunterVMState = ref object of BaseVMState
headers: Table[BlockNumber, BlockHeader]
proc hash*(x: UInt256): Hash =
result = hash(x.toBytesBE)
proc new(T: type HunterVMState; parent, header: BlockHeader, com: CommonRef): T =
new result
result.init(parent, header, com)
result.headers = Table[BlockNumber, BlockHeader]()
method getAncestorHash*(vmState: HunterVMState, blockNumber: BlockNumber): Hash256 =
if blockNumber in vmState.headers:
result = vmState.headers[blockNumber].hash
else:
let data = requestHeader(blockNumber)
let header = parseBlockHeader(data)
result = header.hash
vmState.headers[blockNumber] = header
proc putAncestorsIntoDB(vmState: HunterVMState, db: CoreDbRef) =
for header in vmState.headers.values:
db.addBlockNumberToHashLookup(header)
proc huntProblematicBlock(blockNumber: UInt256): Result[void, string] =
let
# prepare needed state from previous block
parentNumber = blockNumber - 1
thisBlock = requestBlock(blockNumber)
parentBlock = requestBlock(parentNumber)
memoryDB = prepareBlockEnv(parentBlock.header, thisBlock)
# try to execute current block
com = CommonRef.new(memoryDB)
discard com.db.setHead(parentBlock.header, true)
let transaction = memoryDB.beginTransaction()
defer: transaction.dispose()
let
vmState = HunterVMState.new(parentBlock.header, thisBlock.header, com)
validationResult = vmState.processBlock(thisBlock.header, thisBlock.body)
if validationResult.isErr():
transaction.rollback()
putAncestorsIntoDB(vmState, com.db)
vmState.dumpDebuggingMetaData(thisBlock.header, thisBlock.body, false)
validationResult
proc main() {.used.} =
let conf = getConfiguration()
if conf.head == 0.u256:
echo "please specify the starting block with `--head:blockNumber`"
quit(QuitFailure)
if conf.maxBlocks == 0:
echo "please specify the number of problematic blocks you want to hunt with `--maxBlocks:number`"
quit(QuitFailure)
var
problematicBlocks = newSeq[UInt256]()
blockNumber = conf.head
while true:
echo blockNumber
if huntProblematicBlock(blockNumber).isErr:
echo "shot down problematic block: ", blockNumber
problematicBlocks.add blockNumber
blockNumber = blockNumber + 1
if problematicBlocks.len >= conf.maxBlocks:
echo "Problematic blocks: ", problematicBlocks
break
when isMainModule:
var message: string
## Processing command line arguments
if processArguments(message) != Success:
echo message
quit(QuitFailure)
else:
if len(message) > 0:
echo message
quit(QuitSuccess)
try:
main()
except:
echo getCurrentExceptionMsg()

View File

@ -1,125 +0,0 @@
# Nimbus
# Copyright (c) 2020-2024 Status Research & Development GmbH
# 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.
# use this module to quickly populate db with data from geth/parity
import
std/os,
chronicles,
../nimbus/errors,
../nimbus/core/chain,
../nimbus/common,
../nimbus/db/opts,
../nimbus/db/core_db/persistent,
configuration # must be late (compilation annoyance)
when defined(graphql):
import graphql_downloader
else:
import downloader
template persistToDb(db: CoreDbRef, body: untyped) =
block: body
proc contains(kvt: CoreDbKvtRef; key: openArray[byte]): bool =
kvt.hasKeyRc(key).expect "valid bool"
proc main() {.used.} =
# 97 block with uncles
# 46147 block with first transaction
# 46400 block with transaction
# 46402 block with first contract: failed
# 47205 block with first success contract
# 48712 block with 5 transactions
# 48915 block with contract
# 49018 first problematic block
# 49439 first block with contract call
# 52029 first block with receipts logs
# 66407 failed transaction
let conf = configuration.getConfiguration()
let com = CommonRef.new(
newCoreDbRef(DefaultDbPersistent, conf.dataDir, DbOptions.init()),
conf.netId, networkParams(conf.netId))
# move head to block number ...
if conf.head != 0'u64:
var parentBlock = requestBlock(conf.head, { DownloadAndValidate })
discard com.db.setHead(parentBlock.header)
let kvt = com.db.ctx.getKvt()
var head = com.db.getCanonicalHead()
var blockNumber = head.number + 1
var chain = newChain(com)
let numBlocksToCommit = conf.numCommits
var blocks = newSeqOfCap[EthBlock](numBlocksToCommit)
var one = 1'u64
var numBlocks = 0
var counter = 0
var retryCount = 0
while true:
var thisBlock: downloader.Block
try:
thisBlock = requestBlock(blockNumber, { DownloadAndValidate })
except CatchableError as e:
if retryCount < 3:
warn "Unable to get block data via JSON-RPC API", error = e.msg
inc retryCount
sleep(1000)
continue
else:
raise e
blocks.add EthBlock.init(thisBlock.header, thisBlock.body)
info "REQUEST HEADER", blockNumber=blockNumber, txs=thisBlock.body.transactions.len
inc numBlocks
blockNumber += one
if numBlocks == numBlocksToCommit:
persistToDb(com.db):
let res = chain.persistBlocks(blocks)
res.isOkOr:
raise newException(ValidationError, "Error when validating blocks: " & res.error)
numBlocks = 0
blocks.setLen(0)
inc counter
if conf.maxBlocks != 0 and counter >= conf.maxBlocks:
break
if numBlocks > 0:
persistToDb(com.db):
let res = chain.persistBlocks(blocks)
res.isOkOr:
raise newException(ValidationError, "Error when validating blocks: " & res.error)
when isMainModule:
var message: string
## Processing command line arguments
if configuration.processArguments(message) != Success:
if len(message) > 0:
echo message
echo "Usage: persist --datadir=<DATA_DIR> --maxblocks=<MAX_BLOCKS> --head=<HEAD> --numcommits=<NUM_COMMITS> --netid=<NETWORK_ID>"
quit(QuitFailure)
else:
if len(message) > 0:
echo message
quit(QuitSuccess)
try:
main()
except CatchableError:
echo getCurrentExceptionMsg()

View File

@ -1,74 +0,0 @@
# Nimbus
# Copyright (c) 2020-2024 Status Research & Development GmbH
# 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.
import
std/[json, strutils, os],
downloader,
../nimbus/tracer, prestate,
eth/common, premixcore
proc generateGethData(thisBlock: Block, blockNumber: BlockNumber, accounts: JsonNode): JsonNode =
let
receipts = toJson(thisBlock.receipts)
let geth = %{
"blockNumber": %blockNumber.toHex,
"txTraces": thisBlock.traces,
"receipts": receipts,
"block": thisBlock.jsonData,
"accounts": accounts
}
result = geth
proc printDebugInstruction(blockNumber: BlockNumber) =
var text = """
Successfully created debugging environment for block $1.
You can continue to find nimbus EVM bug by viewing premix report page `./index.html`.
After that you can try to debug that single block using `nim c -r debug block$1.json` command.
Happy bug hunting
""" % [$blockNumber]
echo text
proc main() =
if paramCount() == 0:
echo "usage: premix debugxxx.json"
quit(QuitFailure)
try:
let
nimbus = json.parseFile(paramStr(1))
blockNumberHex = nimbus["blockNumber"].getStr()
blockNumber = parseHexInt(blockNumberHex).uint64
thisBlock = requestBlock(blockNumber, {DownloadReceipts, DownloadTxTrace})
accounts = requestPostState(thisBlock)
geth = generateGethData(thisBlock, blockNumber, accounts)
parentNumber = blockNumber - 1
parentBlock = requestBlock(parentNumber)
processNimbusData(nimbus)
# premix data goes to report page
generatePremixData(nimbus, geth)
# prestate data goes to debug tool and contains data
# needed to execute single block
generatePrestate(
nimbus, geth, blockNumber, parentBlock.header,
EthBlock.init(thisBlock.header, thisBlock.body))
printDebugInstruction(blockNumber)
except CatchableError:
echo getCurrentExceptionMsg()
main()

View File

@ -1,184 +0,0 @@
# Nimbus
# Copyright (c) 2020-2024 Status Research & Development GmbH
# 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.
import
json, strutils, os,
chronicles, eth/common, eth/common/transaction_utils,
../nimbus/transaction, ../nimbus/launcher,
./js_tracer, ./parser, ./downloader
proc fakeAlloc(n: JsonNode) =
const
chunk = repeat('0', 64)
for i in 1 ..< n.len:
if not n[i].hasKey("memory"): return
let
prevMem = n[i-1]["memory"]
currMem = n[i]["memory"]
prevPc = n[i-1]["pc"].getInt()
currPc = n[i]["pc"].getInt()
if currMem.len > prevMem.len and prevPc == currPc - 1:
let diff = currMem.len - prevMem.len
for _ in 0 ..< diff:
prevMem.add %chunk
proc updateAccount*(a, b: JsonNode) =
if b.hasKey("name"):
a["name"] = newJString(b["name"].getStr)
a["balance"] = newJString(b["balance"].getStr)
a["nonce"] = newJString(b["nonce"].getStr)
a["code"] = newJString(b["code"].getStr)
var storage = a["storage"]
for k, v in b["storage"]:
storage[k] = newJString(v.getStr)
a["storageRoot"] = newJString(b["storageRoot"].getStr)
a["codeHash"] = newJString(b["codeHash"].getStr)
proc copyAccount*(acc: JsonNode): JsonNode =
result = newJObject()
result["storage"] = newJObject()
updateAccount(result, acc)
proc removePostStateDup*(postState: JsonNode): JsonNode =
var accounts = newJObject()
for acc in postState:
let address = acc["address"].getStr
if accounts.hasKey(address):
updateAccount(accounts[address], acc)
else:
accounts[address] = copyAccount(acc)
accounts
proc processNimbusData*(nimbus: JsonNode) =
# remove duplicate accounts with same address
# and only take newest one
let postState = nimbus["stateDump"]["after"]
nimbus["stateDump"]["after"] = removePostStateDup(postState)
let txTraces = nimbus["txTraces"]
for trace in txTraces:
trace["structLogs"].fakeAlloc()
proc generatePremixData*(nimbus, geth: JsonNode) =
var premixData = %{
"nimbus": nimbus,
"geth": geth
}
var data = "var premixData = " & premixData.pretty & "\n"
writeFile(getFileDir("index.html") / "premixData.js", data)
proc hasInternalTx(tx: Transaction, blockNumber: BlockNumber, sender: EthAddress): bool =
let
number = %(blockNumber.prefixHex)
recipient = tx.getRecipient(sender)
code = request("eth_getCode", %[%recipient.prefixHex, number])
recipientHasCode = code.getStr.len > 2 # "0x"
if tx.contractCreation:
return recipientHasCode or tx.payload.len > 0
recipientHasCode
proc jsonTracer(tracer: string): JsonNode =
result = %{ "tracer": %tracer }
proc requestInternalTx(txHash, tracer: JsonNode): JsonNode =
let txTrace = request("debug_traceTransaction", %[txHash, tracer])
if txTrace.kind == JNull:
error "requested postState not available", txHash=txHash
raise newException(ValueError, "Error when retrieving transaction postState")
result = txTrace
proc requestAccount*(premix: JsonNode, blockNumber: BlockNumber, address: EthAddress) =
let
number = %(blockNumber.prefixHex)
address = address.prefixHex
proof = request("eth_getProof", %[%address, %[], number])
let account = %{
"address": %address,
"codeHash": proof["codeHash"],
"storageRoot": proof["storageHash"],
"balance": proof["balance"],
"nonce": proof["nonce"],
"code": newJString("0x"),
"storage": newJObject(),
"accountProof": proof["accountProof"],
"storageProof": proof["storageProof"]
}
premix.add account
proc padding(x: string): JsonNode =
let val = x.substr(2)
let pad = repeat('0', 64 - val.len)
result = newJString("0x" & pad & val)
proc updateAccount*(address: string, account: JsonNode, blockNumber: BlockNumber) =
let number = %(blockNumber.prefixHex)
var storage = newJArray()
for k, _ in account["storage"]:
storage.add %k
let proof = request("eth_getProof", %[%address, storage, number])
account["address"] = %address
account["codeHash"] = proof["codeHash"]
account["storageRoot"] = proof["storageHash"]
account["nonce"] = proof["nonce"]
account["balance"] = proof["balance"]
account["accountProof"]= proof["accountProof"]
account["storageProof"]= proof["storageProof"]
for x in proof["storageProof"]:
x["value"] = padding(x["value"].getStr())
account["storage"][x["key"].getStr] = x["value"]
proc requestPostState*(premix, n: JsonNode, blockNumber: BlockNumber) =
type
TxKind {.pure.} = enum
Regular
ContractCreation
ContractCall
let txs = n["transactions"]
if txs.len == 0: return
let tracer = jsonTracer(postStateTracer)
for t in txs:
var txKind = TxKind.Regular
let tx = parseTransaction(t)
let sender = tx.recoverSender().valueOr:
raise (ref ValueError)(msg: "Invalid tx signature")
if tx.contractCreation: txKind = TxKind.ContractCreation
if hasInternalTx(tx, blockNumber, sender):
let txTrace = requestInternalTx(t["hash"], tracer)
for address, account in txTrace:
updateAccount(address, account, blockNumber)
premix.add account
if not tx.contractCreation: txKind = TxKind.ContractCall
else:
premix.requestAccount(blockNumber, tx.getRecipient(sender))
premix.requestAccount(blockNumber, sender)
t["txKind"] = %($txKind)
proc requestPostState*(thisBlock: Block): JsonNode =
let blockNumber = thisBlock.header.number
var premix = newJArray()
premix.requestPostState(thisBlock.jsonData, blockNumber)
premix.requestAccount(blockNumber, thisBlock.header.coinbase)
for uncle in thisBlock.body.uncles:
premix.requestAccount(blockNumber, uncle.coinbase)
removePostStateDup(premix)

View File

@ -1,46 +0,0 @@
# Nimbus
# Copyright (c) 2020-2024 Status Research & Development GmbH
# 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.
import
std/strutils,
json, stew/byteutils,
results,
../nimbus/db/[core_db, storage_types], eth/[rlp, common],
../nimbus/tracer
proc generatePrestate*(nimbus, geth: JsonNode, blockNumber: BlockNumber, parent: BlockHeader, blk: EthBlock) =
template header: BlockHeader = blk.header
let
state = nimbus["state"]
headerHash = rlpHash(header)
chainDB = newCoreDbRef(DefaultDbMemory)
kvt = chainDB.ctx.getKvt()
discard chainDB.setHead(parent, true)
chainDB.persistTransactions(blockNumber, header.txRoot, blk.transactions)
discard chainDB.persistUncles(blk.uncles)
kvt.put(genericHashKey(headerHash).toOpenArray, rlp.encode(header)).isOkOr:
raiseAssert "generatePrestate(): put() failed " & $$error
chainDB.addBlockNumberToHashLookup(header.number, headerHash)
for k, v in state:
let key = hexToSeqByte(k)
let value = hexToSeqByte(v.getStr())
kvt.put(key, value).isOkOr:
raiseAssert "generatePrestate(): put() (loop) failed " & $$error
var metaData = %{
"blockNumber": %blockNumber.toHex,
"geth": geth
}
metaData.dumpMemoryDB(chainDB)
writeFile("block" & $blockNumber & ".json", metaData.pretty())

View File

@ -1,153 +0,0 @@
# Premix
> Premix is **pre**mium gasoline **mix**ed with lubricant oil and it is
used in two-stroke internal combustion engines. It tends to produce a lot of
smoke.
This Premix is a block validation debugging tool for the Nimbus Ethereum
client. Premix will query transaction execution steps from other Ethereum
clients and compare them with those generated by Nimbus. It will then produce a
web page to present comparison results that can be inspected by the developer
to pinpoint the faulty instruction.
Premix will also produce a test case for the specific problematic transaction,
complete with a database snapshot to execute transaction validation in
isolation. This test case can then be integrated with the Nimbus project's test
suite.
![screenshot](assets/images/premix_screenshot.png)
## Requirements
Before you can use the Premix debugging tool there are several things you need
to prepare. The first requirement is a recent version of `geth` installed from
[source](https://github.com/ethereum/go-ethereum/releases) or
[binary](https://ethereum.github.io/go-ethereum/downloads/). The minimum
required version is 1.8.18. Beware that version 1.8.x contains bugs in
transaction tracer, upgrade it to 1.9.x soon after it has been released.
Afterwards, you can run it with this command:
```bash
geth --rpc --rpcapi eth,debug --syncmode full --gcmode=archive
```
You need to run it until it fully syncs past the problematic block you want to
debug (you might need to do it on an empty db, because some geth versions will
keep on doing a fast sync if that's what was done before). After that, you can
stop it by pressing `CTRL-C` and rerun it with the additional flag `--maxpeers
0` if you want it to stop syncing
- or just let it run as is if you want to keep syncing.
The next requirement is building Nimbus and Premix:
```bash
# in the top-level directory:
make
```
After that, you can run Nimbus with this command:
```bash
./build/nimbus --prune:archive --port:30304
```
Nimbus will try to sync up to the problematic block, then stop and execute
Premix which will then load a report page in your default browser. If it fails
to do that, you can see the report page by manually opening
`premix/index.html`.
In your browser, you can explore the tracing result and find where the problem is.
## Tools
### Premix
Premix is the main debugging tool. It produces reports that can be viewed in
a browser and serialised debug data that can be consumed by the `debug` tool.
Premix consumes data produced by either `nimbus`, `persist`, or `dumper`.
You can run it manually using this command:
```bash
./build/premix debug*.json
```
### Persist
Because the Nimbus P2P layer still contains bugs, you may become impatient when
trying to sync blocks. In the `./premix` directory, you can find a `persist`
tool. It will help you sync relatively quicker because it will bypass the P2P
layer and download blocks from `geth` via `rpc-api`.
When it encounters a problematic block during syncing, it will stop and produce
debugging data just like Nimbus does.
```bash
./build/persist [--dataDir:your_database_directory] [--head: blockNumber] [--maxBlocks: number] [--numCommits: number]
```
### Debug
In the same `./premix` directory you'll find the `debug` tool that you can use
to process previously generated debugging info in order to work with one block
and one transaction at a time instead of multiple confusing blocks and
transactions.
```bash
./build/debug block*.json
```
where `block*.json` contains the database snapshot needed to debug a single
block produced by the Premix tool.
### Dumper
Dumper was designed specifically to produce debugging data that can be further
processed by Premix from information already stored in database. It will create
tracing information for a single block if that block has been already
persisted.
If you want to generate debugging data, it's better to use the Persist tool.
The data generated by Dumper is usually used to debug Premix features in
general and the report page logic in particular.
```bash
# usage:
./build/dumper [--datadir:your_path] --head:blockNumber
```
### Hunter
Hunter's purpose is to track down problematic blocks and create debugging info
associated with them. It will not access your on-disk database, because it has
its own prestate construction code.
Hunter will download all it needs from geth, just make sure your geth version
is at least 1.8.18.
Hunter depends on
`eth_getProof`[(EIP1186)](https://github.com/ethereum/EIPs/issues/1186). Make
sure your installed `geth` supports this functionality (older versions don't
have this implemented).
```bash
# usage:
./build/hunter --head:blockNumber --maxBlocks:number
```
`blockNumber` is the starting block where the hunt begins.
`maxBlocks` is the number of problematic blocks you want to capture before
stopping the hunt.
### Regress
Regress is an offline block validation tool. It will not download block
information from anywhere like Persist tool. Regress will validate your
already persisted block in database. It will try to find any regression
introduced either by bugfixing or refactoring.
```bash
# usage:
./build/regress [--dataDir:your_db_path] --head:blockNumber
```

View File

@ -1,87 +0,0 @@
# Nimbus
# Copyright (c) 2020-2024 Status Research & Development GmbH
# 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.
import
chronicles,
../nimbus/[evm/state, evm/types],
../nimbus/core/executor,
../nimbus/common/common,
../nimbus/db/opts,
../nimbus/db/core_db/persistent,
configuration # must be late (compilation annoyance)
const
numBlocks = 256
proc validateBlock(com: CommonRef, blockNumber: BlockNumber): BlockNumber =
var
parentNumber = blockNumber - 1
parent = com.db.getBlockHeader(parentNumber)
blocks = newSeq[EthBlock](numBlocks)
for i in 0 ..< numBlocks:
blocks[i] = com.db.getEthBlock(blockNumber + i.BlockNumber)
let transaction = com.db.ctx.newTransaction()
defer: transaction.dispose()
for i in 0 ..< numBlocks:
stdout.write blockNumber + i.BlockNumber
stdout.write "\r"
let
vmState = BaseVMState.new(parent, blocks[i].header, com)
validationResult = vmState.processBlock(blocks[i])
if validationResult.isErr:
error "block validation error",
err = validationResult.error(), blockNumber = blockNumber + i.BlockNumber
parent = blocks[i].header
transaction.rollback()
result = blockNumber + numBlocks.BlockNumber
proc main() {.used.} =
let
conf = getConfiguration()
com = CommonRef.new(newCoreDbRef(
DefaultDbPersistent, conf.dataDir, DbOptions.init()))
# move head to block number ...
if conf.head == 0'u64:
raise newException(ValueError, "please set block number with --head: blockNumber")
var counter = 0
var blockNumber = conf.head
while true:
blockNumber = com.validateBlock(blockNumber)
inc counter
if conf.maxBlocks != 0 and counter >= conf.maxBlocks:
break
when isMainModule:
var message: string
## Processing command line arguments
if processArguments(message) != Success:
echo message
quit(QuitFailure)
else:
if len(message) > 0:
echo message
quit(QuitSuccess)
try:
main()
except:
echo getCurrentExceptionMsg()

View File

@ -12,18 +12,12 @@
{.warning[UnusedImport]: off.} {.warning[UnusedImport]: off.}
import import
#../premix/premix, # -- currently disabled (no tracer at the moment)
#../premix/persist, # -- ditto
#../premix/debug, # -- ditto
#../premix/dumper, # -- ditto
#../premix/hunter, # -- ditto
#../premix/regress, # -- ditto
#./tracerTestGen, # -- ditto #./tracerTestGen, # -- ditto
#./persistBlockTestGen, # -- ditto #./persistBlockTestGen, # -- ditto
../hive_integration/nodocker/rpc/rpc_sim, ../hive_integration/nodocker/rpc/rpc_sim,
../hive_integration/nodocker/consensus/consensus_sim, ../hive_integration/nodocker/consensus/consensus_sim,
#../hive_integration/nodocker/graphql/graphql_sim, # -- does not compile #../hive_integration/nodocker/graphql/graphql_sim, # -- does not compile
#../hive_integration/nodocker/engine/engine_sim, # -- does not compile ../hive_integration/nodocker/engine/engine_sim,
../hive_integration/nodocker/pyspec/pyspec_sim, ../hive_integration/nodocker/pyspec/pyspec_sim,
../tools/t8n/t8n, ../tools/t8n/t8n,
../tools/t8n/t8n_test, ../tools/t8n/t8n_test,