Skip to content

Commit

Permalink
Blockchain debug additions (#3676)
Browse files Browse the repository at this point in the history
* Add debug logging to core blockchain class

* Add debug logging to PoW consensus validation

* Remove logs where errors are thrown

* Update blockchain docs with new debug namespaces

* Put consensus debug namespaces in the same parent space

* Fix spelling mistake

* Fix lint issues

* Optimize out hash recalculation

* Add recomended stats to debug logs

* Rename debug namespaces to be more simple

* Some debug log enhancements

* Fix spelling

---------

Co-authored-by: Holger Drewes <[email protected]>
  • Loading branch information
scorbajio and holgerd77 authored Sep 19, 2024
1 parent 9c16211 commit 4aa4cef
Show file tree
Hide file tree
Showing 4 changed files with 80 additions and 5 deletions.
8 changes: 5 additions & 3 deletions packages/blockchain/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -245,9 +245,11 @@ This library uses the [debug](https://github.com/visionmedia/debug) debugging ut

The following initial logger is currently available:

| Logger | Description |
| ------------------- | ----------------------------------------------------------- |
| `blockchain:clique` | Clique operations like updating the vote and/or signer list |
| Logger | Description |
| ------------------- | ------------------------------------------------------------------------ |
| `blockchain` | Core blockchain operations like when a block or header is put or deleted |
| `blockchain:clique` | Clique consensus operations like updating the vote and/or signer list |
| `blockchain:ethash` | Ethash consensus operations like PoW block or header validation |

The following is an example for a logger run:

Expand Down
50 changes: 49 additions & 1 deletion packages/blockchain/src/blockchain.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,11 +8,13 @@ import {
KECCAK256_RLP,
Lock,
MapDB,
bigIntToHex,
bytesToHex,
bytesToUnprefixedHex,
concatBytes,
equalsBytes,
} from '@ethereumjs/util'
import debugDefault from 'debug'

import { CasperConsensus } from './consensus/casper.js'
import {
Expand All @@ -36,6 +38,7 @@ import type {
import type { HeaderData } from '@ethereumjs/block'
import type { CliqueConfig } from '@ethereumjs/common'
import type { BigIntLike, DB, DBObject, GenesisState } from '@ethereumjs/util'
import type { Debugger } from 'debug'

/**
* Blockchain implementation to create and maintain a valid canonical chain
Expand Down Expand Up @@ -88,6 +91,9 @@ export class Blockchain implements BlockchainInterface {
*/
private _deletedBlocks: Block[] = []

private DEBUG: boolean // Guard for debug logs
private _debug: Debugger

/**
* Creates new Blockchain object.
*
Expand All @@ -99,6 +105,10 @@ export class Blockchain implements BlockchainInterface {
* {@link BlockchainOptions}.
*/
constructor(opts: BlockchainOptions = {}) {
this.DEBUG =
typeof window === 'undefined' ? (process?.env?.DEBUG?.includes('ethjs') ?? false) : false
this._debug = debugDefault('blockchain')

if (opts.common) {
this.common = opts.common
} else {
Expand Down Expand Up @@ -262,6 +272,8 @@ export class Blockchain implements BlockchainInterface {
for (let i = 0; i < blocks.length; i++) {
await this.putBlock(blocks[i])
}

this.DEBUG && this._debug(`put ${blocks.length} blocks`)
}

/**
Expand Down Expand Up @@ -289,6 +301,8 @@ export class Blockchain implements BlockchainInterface {
for (let i = 0; i < headers.length; i++) {
await this.putHeader(headers[i])
}

this.DEBUG && this._debug(`put ${headers.length} headers`)
}

/**
Expand All @@ -312,8 +326,10 @@ export class Blockchain implements BlockchainInterface {
*/

async resetCanonicalHead(canonicalHead: bigint) {
let hash: Uint8Array | undefined
const canonicalHeadHash = (await this.getCanonicalHeadHeader()).hash()
await this.runWithLock<void>(async () => {
const hash = await this.dbManager.numberToHash(canonicalHead)
hash = await this.dbManager.numberToHash(canonicalHead)
if (hash === undefined) {
throw new Error(`no block for ${canonicalHead} found in DB`)
}
Expand All @@ -328,6 +344,17 @@ export class Blockchain implements BlockchainInterface {
})
if (this._deletedBlocks.length > 0) {
this.events.emit('deletedCanonicalBlocks', this._deletedBlocks)
for (const block of this._deletedBlocks)
this.DEBUG &&
this._debug(
`deleted block along head reset: number ${block.header.number} hash ${bytesToHex(block.hash())}`,
)

this.DEBUG &&
this._debug(
`Canonical head set from ${bytesToHex(canonicalHeadHash)} to ${bytesToHex(hash!)} (number ${bigIntToHex(canonicalHead)})`,
)

this._deletedBlocks = []
}
}
Expand Down Expand Up @@ -452,6 +479,8 @@ export class Blockchain implements BlockchainInterface {
await this.dbManager.batch(ops)

await this.consensus?.newBlock(block, commonAncestor, ancestorHeaders)
this.DEBUG &&
this._debug(`put block number=${block.header.number} hash=${bytesToHex(blockHash)}`)
} catch (e) {
// restore head to the previously sane state
this._heads = oldHeads
Expand All @@ -462,6 +491,11 @@ export class Blockchain implements BlockchainInterface {
})
if (this._deletedBlocks.length > 0) {
this.events.emit('deletedCanonicalBlocks', this._deletedBlocks)
for (const block of this._deletedBlocks)
this.DEBUG &&
this._debug(
`delete stale canonical block number=${block.header.number} hash=${bytesToHex(block.hash())}`,
)
this._deletedBlocks = []
}
}
Expand Down Expand Up @@ -829,6 +863,11 @@ export class Blockchain implements BlockchainInterface {

if (this._deletedBlocks.length > 0) {
this.events.emit('deletedCanonicalBlocks', this._deletedBlocks)
for (const block of this._deletedBlocks)
this.DEBUG &&
this._debug(
`delete stale canonical block number=${block.header.number} hash=${blockHash})}`,
)
this._deletedBlocks = []
}
}
Expand Down Expand Up @@ -1018,6 +1057,9 @@ export class Blockchain implements BlockchainInterface {
if (!equalsBytes(header.hash(), newHeader.hash())) {
throw new Error('Failed to find ancient header')
}

this.DEBUG && this._debug(`found common ancestor with hash=${bytesToHex(header.hash())}`)
this.DEBUG && this._debug(`total ancestor headers num=${ancestorHeaders.size}`)
return {
commonAncestor: header,
ancestorHeaders: Array.from(ancestorHeaders),
Expand Down Expand Up @@ -1080,6 +1122,10 @@ export class Blockchain implements BlockchainInterface {

hash = await this.safeNumberToHash(blockNumber)
}

this.DEBUG &&
this._deletedBlocks.length > 0 &&
this._debug(`deleted ${this._deletedBlocks.length} stale canonical blocks in total`)
} catch (e) {
// Ensure that if this method throws, `_deletedBlocks` is reset to the empty array
this._deletedBlocks = []
Expand Down Expand Up @@ -1162,6 +1208,8 @@ export class Blockchain implements BlockchainInterface {
if (staleHeadBlock) {
this._headBlockHash = currentCanonicalHash
}

this.DEBUG && this._debug(`stale heads found num=${staleHeads.length}`)
}

/* Helper functions */
Expand Down
18 changes: 18 additions & 0 deletions packages/blockchain/src/consensus/ethash.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,11 @@
import { ConsensusAlgorithm } from '@ethereumjs/common'
import { bytesToHex } from '@ethereumjs/util'
import debugDefault from 'debug'

import type { Blockchain } from '../index.js'
import type { Consensus, ConsensusOptions } from '../types.js'
import type { Block, BlockHeader } from '@ethereumjs/block'
import type { Debugger } from 'debug'

type MinimalEthashInterface = {
cacheDB?: any
Expand All @@ -17,7 +20,14 @@ export class EthashConsensus implements Consensus {
algorithm: ConsensusAlgorithm
_ethash: MinimalEthashInterface

private DEBUG: boolean // Guard for debug logs
private _debug: Debugger

constructor(ethash: MinimalEthashInterface) {
this.DEBUG =
typeof window === 'undefined' ? (process?.env?.DEBUG?.includes('ethjs') ?? false) : false
this._debug = debugDefault('blockchain:ethash')

this.algorithm = ConsensusAlgorithm.Ethash
this._ethash = ethash
}
Expand All @@ -27,6 +37,10 @@ export class EthashConsensus implements Consensus {
if (!valid) {
throw new Error('invalid POW')
}
this.DEBUG &&
this._debug(
`valid PoW consensus block: number ${block.header.number} hash ${bytesToHex(block.hash())}`,
)
}

/**
Expand All @@ -41,6 +55,10 @@ export class EthashConsensus implements Consensus {
if (header.ethashCanonicalDifficulty(parentHeader) !== header.difficulty) {
throw new Error(`invalid difficulty ${header.errorStr()}`)
}
this.DEBUG &&
this._debug(
`valid difficulty header: number ${header.number} difficulty ${header.difficulty} parentHash ${bytesToHex(header.parentHash)}`,
)
}

public async genesisInit(): Promise<void> {}
Expand Down
9 changes: 8 additions & 1 deletion packages/blockchain/src/constructors.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { createBlock } from '@ethereumjs/block'
import { BIGINT_0, equalsBytes } from '@ethereumjs/util'
import { BIGINT_0, bytesToHex, equalsBytes } from '@ethereumjs/util'
import debugDefault from 'debug'

import {
Blockchain,
Expand All @@ -14,6 +15,10 @@ import type { BlockchainOptions, DBOp } from './index.js'
import type { BlockData } from '@ethereumjs/block'
import type { Chain } from '@ethereumjs/common'

const DEBUG =
typeof window === 'undefined' ? (process?.env?.DEBUG?.includes('ethjs') ?? false) : false
const debug = debugDefault('blockchain')

export async function createBlockchain(opts: BlockchainOptions = {}) {
const blockchain = new Blockchain(opts)

Expand Down Expand Up @@ -81,6 +86,8 @@ export async function createBlockchain(opts: BlockchainOptions = {}) {
await blockchain.checkAndTransitionHardForkByNumber(latestHeader.number, latestHeader.timestamp)
}

DEBUG && debug(`genesis block initialized with hash ${bytesToHex(genesisHash!)}`)

return blockchain
}

Expand Down

0 comments on commit 4aa4cef

Please sign in to comment.