Slightly reworked ChainDB interface to more consistent

This commit is contained in:
Yuriy Glukhov 2018-08-03 14:56:49 +03:00
parent db62ef132e
commit e4929d4f60

View File

@ -30,36 +30,55 @@ proc newBaseChainDB*(db: TrieDatabaseRef): BaseChainDB =
proc `$`*(db: BaseChainDB): string = proc `$`*(db: BaseChainDB): string =
result = "BaseChainDB" result = "BaseChainDB"
proc getBlockHeaderByHash*(self: BaseChainDB; blockHash: Hash256): BlockHeader = proc getBlockHeader*(self: BaseChainDB; blockHash: Hash256, output: var BlockHeader): bool =
## Returns the requested block header as specified by block hash.
##
## Raises BlockNotFound if it is not present in the db.
try: try:
let blk = self.db.get(genericHashKey(blockHash).toOpenArray).toRange let blk = self.db.get(genericHashKey(blockHash).toOpenArray).toRange
return decode(blk, BlockHeader) if blk.len != 0:
output = rlp.decode(blk, BlockHeader)
result = true
except KeyError: except KeyError:
discard
proc getBlockHeader*(self: BaseChainDB, blockHash: Hash256): BlockHeader =
## Returns the requested block header as specified by block hash.
##
## Raises BlockNotFound if it is not present in the db.
if not self.getBlockHeader(blockHash, result):
raise newException(BlockNotFound, "No block with hash " & blockHash.data.toHex) raise newException(BlockNotFound, "No block with hash " & blockHash.data.toHex)
proc getHash(self: BaseChainDB, key: DbKey): Hash256 {.inline.} = proc getHash(self: BaseChainDB, key: DbKey, output: var Hash256): bool {.inline.} =
rlp.decode(self.db.get(key.toOpenArray).toRange, Hash256) try:
output = rlp.decode(self.db.get(key.toOpenArray).toRange, Hash256)
result = true
except KeyError:
discard
proc getCanonicalHead*(self: BaseChainDB): BlockHeader = proc getCanonicalHead*(self: BaseChainDB): BlockHeader =
let k = canonicalHeadHashKey() var headHash: Hash256
if k.toOpenArray notin self.db: if not self.getHash(canonicalHeadHashKey(), headHash) or
not self.getBlockHeader(headHash, result):
raise newException(CanonicalHeadNotFound, raise newException(CanonicalHeadNotFound,
"No canonical head set for this chain") "No canonical head set for this chain")
return self.getBlockHeaderByHash(self.getHash(k))
proc lookupBlockHash*(self: BaseChainDB; n: BlockNumber): Hash256 {.inline.} = proc getBlockHash*(self: BaseChainDB, n: BlockNumber, output: var Hash256): bool {.inline.} =
## Return the block hash for the given block number. ## Return the block hash for the given block number.
self.getHash(blockNumberToHashKey(n)) self.getHash(blockNumberToHashKey(n), output)
proc getCanonicalBlockHeaderByNumber*(self: BaseChainDB; n: BlockNumber): BlockHeader = proc getBlockHash*(self: BaseChainDB, n: BlockNumber): Hash256 {.inline.} =
## Returns the block header with the given number in the canonical chain. ## Return the block hash for the given block number.
## if not self.getHash(blockNumberToHashKey(n), result):
## Raises BlockNotFound if there's no block header with the given number in the raise newException(BlockNotFound, "No block hash for number " & $n)
## canonical chain.
self.getBlockHeaderByHash(self.lookupBlockHash(n)) proc getBlockHeader*(self: BaseChainDB; n: BlockNumber, output: var BlockHeader): bool =
## Returns the block header with the given number in the canonical chain.
var blockHash: Hash256
if self.getBlockHash(n, blockHash):
result = self.getBlockHeader(blockHash, output)
proc getBlockHeader*(self: BaseChainDB; n: BlockNumber): BlockHeader =
## Returns the block header with the given number in the canonical chain.
## Raises BlockNotFound error if the block is not in the DB.
self.getBlockHeader(self.getBlockHash(n))
proc getScore*(self: BaseChainDB; blockHash: Hash256): int = proc getScore*(self: BaseChainDB; blockHash: Hash256): int =
rlp.decode(self.db.get(blockHashToScoreKey(blockHash).toOpenArray).toRange, int) rlp.decode(self.db.get(blockHashToScoreKey(blockHash).toOpenArray).toRange, int)
@ -68,20 +87,17 @@ iterator findNewAncestors(self: BaseChainDB; header: BlockHeader): BlockHeader =
## Returns the chain leading up from the given header until the first ancestor it has in ## Returns the chain leading up from the given header until the first ancestor it has in
## common with our canonical chain. ## common with our canonical chain.
var h = header var h = header
var orig: BlockHeader
while true: while true:
try: if self.getBlockHeader(h.blockNumber, orig) and orig.hash == h.hash:
let orig = self.getCanonicalBlockHeaderByNumber(h.blockNumber) break
if orig.hash == h.hash:
break
except BlockNotFound:
discard
yield h yield h
if h.parentHash == GENESIS_PARENT_HASH: if h.parentHash == GENESIS_PARENT_HASH:
break break
else: else:
h = self.getBlockHeaderByHash(h.parentHash) h = self.getBlockHeader(h.parentHash)
proc addBlockNumberToHashLookup(self: BaseChainDB; header: BlockHeader) = proc addBlockNumberToHashLookup(self: BaseChainDB; header: BlockHeader) =
self.db.put(blockNumberToHashKey(header.blockNumber).toOpenArray, self.db.put(blockNumberToHashKey(header.blockNumber).toOpenArray,
@ -104,19 +120,16 @@ proc removeTransactionFromCanonicalChain(self: BaseChainDB, transactionHash: Has
proc setAsCanonicalChainHead(self: BaseChainDB; headerHash: Hash256): seq[BlockHeader] = proc setAsCanonicalChainHead(self: BaseChainDB; headerHash: Hash256): seq[BlockHeader] =
## Sets the header as the canonical chain HEAD. ## Sets the header as the canonical chain HEAD.
let header = self.getBlockHeader(headerHash)
let header = self.getBlockHeaderByHash(headerHash)
var newCanonicalHeaders = sequtils.toSeq(findNewAncestors(self, header)) var newCanonicalHeaders = sequtils.toSeq(findNewAncestors(self, header))
reverse(newCanonicalHeaders) reverse(newCanonicalHeaders)
for h in newCanonicalHeaders: for h in newCanonicalHeaders:
var oldHash: Hash256 var oldHash: Hash256
try: if not self.getBlockHash(h.blockNumber, oldHash):
oldHash = self.lookupBlockHash(h.blockNumber)
except BlockNotFound:
break break
let oldHeader = self.getBlockHeaderByHash(oldHash) let oldHeader = self.getBlockHeader(oldHash)
for txHash in self.getBlockTransactionHashes(oldHeader): for txHash in self.getBlockTransactionHashes(oldHeader):
self.removeTransactionFromCanonicalChain(txHash) self.removeTransactionFromCanonicalChain(txHash)
# TODO re-add txn to internal pending pool (only if local sender) # TODO re-add txn to internal pending pool (only if local sender)
@ -232,25 +245,41 @@ proc persistBlockToDb*(self: BaseChainDB; blk: Block) =
proc getStateDb*(self: BaseChainDB; stateRoot: Hash256; readOnly: bool = false): AccountStateDB = proc getStateDb*(self: BaseChainDB; stateRoot: Hash256; readOnly: bool = false): AccountStateDB =
result = newAccountStateDB(self.db, stateRoot) result = newAccountStateDB(self.db, stateRoot)
method genesisHash*(db: BaseChainDB): KeccakHash = method genesisHash*(db: BaseChainDB): KeccakHash =
db.lookupBlockHash(0.toBlockNumber) db.getBlockHash(0.toBlockNumber)
method getBlockHeader*(db: BaseChainDB, b: HashOrNum): BlockHeaderRef = method getBlockHeader*(db: BaseChainDB, b: HashOrNum): BlockHeaderRef =
result.new() var h: BlockHeader
case b.isHash var ok = case b.isHash
of true: of true:
result[] = db.getBlockHeaderByHash(b.hash) db.getBlockHeader(b.hash, h)
else: else:
result[] = db.getBlockHeaderByHash(db.lookupBlockHash(b.number)) db.getBlockHeader(b.number, h)
if ok:
result.new()
result[] = h
method getBestBlockHeader*(self: BaseChainDB): BlockHeaderRef = method getBestBlockHeader*(self: BaseChainDB): BlockHeaderRef =
result.new() result.new()
result[] = self.getCanonicalHead() result[] = self.getCanonicalHead()
# method getSuccessorHeader*(db: BaseChainDB, method getSuccessorHeader*(db: BaseChainDB, h: BlockHeader): BlockHeaderRef =
# h: BlockHeader): BlockHeaderRef = let n = h.blockNumber + 1
# notImplemented var r: BlockHeader
if db.getBlockHeader(n, r):
result.new()
result[] = r
# method getBlockBody*(db: BaseChainDB, blockHash: KeccakHash): BlockBodyRef = method getBlockBody*(db: BaseChainDB, blockHash: KeccakHash): BlockBodyRef =
# notImplemented result = nil
# Deprecated:
proc getBlockHeaderByHash*(self: BaseChainDB; blockHash: Hash256): BlockHeader {.deprecated.} =
self.getBlockHeader(blockHash)
proc lookupBlockHash*(self: BaseChainDB; n: BlockNumber): Hash256 {.deprecated.} =
self.getBlockHash(n)
proc getCanonicalBlockHeaderByNumber*(self: BaseChainDB; n: BlockNumber): BlockHeader {.deprecated.} =
self.getBlockHeader(n)