258 lines
8.1 KiB
Nim
Raw Normal View History

# nimbus_verified_proxy
# Copyright (c) 2022-2024 Status Research & Development GmbH
# Licensed and distributed under either of
# * MIT license (license terms in the root directory or at https://opensource.org/licenses/MIT).
# * Apache v2 license (license terms in the root directory or at https://www.apache.org/licenses/LICENSE-2.0).
# at your option. This file may not be copied, modified, or distributed except according to those terms.
{.push raises: [].}
import
std/strutils,
2024-05-30 14:54:03 +02:00
results,
chronicles,
json_rpc/[rpcproxy, rpcserver, rpcclient],
eth/common/accounts,
web3/[primitives, eth_api_types, eth_api],
../validate_proof,
../block_cache
logScope:
topics = "verified_proxy"
type
VerifiedRpcProxy* = ref object
proxy: RpcProxy
blockCache: BlockCache
chainId: Quantity
QuantityTagKind = enum
LatestBlock
BlockNumber
BlockTag = eth_api_types.RtBlockIdentifier
QuantityTag = object
case kind: QuantityTagKind
of LatestBlock:
discard
of BlockNumber:
blockNumber: Quantity
func parseQuantityTag(blockTag: BlockTag): Result[QuantityTag, string] =
if blockTag.kind == bidAlias:
let tag = blockTag.alias.toLowerAscii
case tag
of "latest":
return ok(QuantityTag(kind: LatestBlock))
else:
return err("Unsupported blockTag: " & tag)
else:
let quantity = blockTag.number
return ok(QuantityTag(kind: BlockNumber, blockNumber: quantity))
template checkPreconditions(proxy: VerifiedRpcProxy) =
if proxy.blockCache.isEmpty():
raise newException(ValueError, "Syncing")
template rpcClient(lcProxy: VerifiedRpcProxy): RpcClient =
lcProxy.proxy.getClient()
proc getBlockByTag(
proxy: VerifiedRpcProxy, quantityTag: BlockTag
): results.Opt[BlockObject] {.raises: [ValueError].} =
checkPreconditions(proxy)
let tag = parseQuantityTag(quantityTag).valueOr:
raise newException(ValueError, error)
case tag.kind
of LatestBlock:
# this will always return some block, as we always checkPreconditions
proxy.blockCache.latest
of BlockNumber:
proxy.blockCache.getByNumber(tag.blockNumber)
proc getBlockByTagOrThrow(
proxy: VerifiedRpcProxy, quantityTag: BlockTag
): BlockObject {.raises: [ValueError].} =
getBlockByTag(proxy, quantityTag).valueOr:
raise newException(ValueError, "No block stored for given tag " & $quantityTag)
proc installEthApiHandlers*(lcProxy: VerifiedRpcProxy) =
lcProxy.proxy.rpc("eth_chainId") do() -> Quantity:
lcProxy.chainId
lcProxy.proxy.rpc("eth_blockNumber") do() -> Quantity:
## Returns the number of the most recent block.
let latest = lcProxy.blockCache.latest.valueOr:
raise (ref ValueError)(msg: "Syncing")
latest.number
lcProxy.proxy.rpc("eth_getBalance") do(
address: Address, quantityTag: BlockTag
) -> UInt256:
# When requesting state for `latest` block number, we need to translate
# `latest` to actual block number as `latest` on proxy and on data provider
# can mean different blocks and ultimatly piece received piece of state
# must by validated against correct state root
let
blk = lcProxy.getBlockByTagOrThrow(quantityTag)
blockNumber = blk.number.uint64
info "Forwarding eth_getBalance call", blockNumber
let
proof = await lcProxy.rpcClient.eth_getProof(address, @[], blockId(blockNumber))
account = getAccountFromProof(
blk.stateRoot, proof.address, proof.balance, proof.nonce, proof.codeHash,
proof.storageHash, proof.accountProof,
).valueOr:
raise newException(ValueError, error)
account.balance
lcProxy.proxy.rpc("eth_getStorageAt") do(
address: Address, slot: UInt256, quantityTag: BlockTag
) -> UInt256:
let
blk = lcProxy.getBlockByTagOrThrow(quantityTag)
blockNumber = blk.number.uint64
info "Forwarding eth_getStorageAt", blockNumber
let proof =
await lcProxy.rpcClient.eth_getProof(address, @[slot], blockId(blockNumber))
getStorageData(blk.stateRoot, slot, proof).valueOr:
raise newException(ValueError, error)
lcProxy.proxy.rpc("eth_getTransactionCount") do(
address: Address, quantityTag: BlockTag
) -> Quantity:
2022-09-28 22:43:10 +02:00
let
blk = lcProxy.getBlockByTagOrThrow(quantityTag)
blockNumber = blk.number.uint64
2022-09-28 22:43:10 +02:00
info "Forwarding eth_getTransactionCount", blockNumber
2022-09-28 22:43:10 +02:00
let
proof = await lcProxy.rpcClient.eth_getProof(address, @[], blockId(blockNumber))
2022-09-28 22:43:10 +02:00
account = getAccountFromProof(
blk.stateRoot, proof.address, proof.balance, proof.nonce, proof.codeHash,
proof.storageHash, proof.accountProof,
).valueOr:
raise newException(ValueError, error)
2022-09-28 22:43:10 +02:00
Quantity(account.nonce)
2022-09-28 22:43:10 +02:00
lcProxy.proxy.rpc("eth_getCode") do(
address: Address, quantityTag: BlockTag
) -> seq[byte]:
2022-11-21 13:28:15 +01:00
let
blk = lcProxy.getBlockByTagOrThrow(quantityTag)
blockNumber = blk.number.uint64
2022-11-21 13:28:15 +01:00
info "Forwarding eth_getCode", blockNumber
2022-11-21 13:28:15 +01:00
let
proof = await lcProxy.rpcClient.eth_getProof(address, @[], blockId(blockNumber))
account = getAccountFromProof(
blk.stateRoot, proof.address, proof.balance, proof.nonce, proof.codeHash,
proof.storageHash, proof.accountProof,
).valueOr:
raise newException(ValueError, error)
2022-11-21 13:28:15 +01:00
if account.codeHash == EMPTY_CODE_HASH:
2022-11-21 13:28:15 +01:00
# account does not have any code, return empty hex data
return @[]
2022-11-21 13:28:15 +01:00
let code = await lcProxy.rpcClient.eth_getCode(address, blockId(blockNumber))
2022-11-21 13:28:15 +01:00
if isValidCode(account, code):
return code
2022-11-21 13:28:15 +01:00
else:
raise newException(
ValueError, "Received code which does not match the account code hash"
)
2022-11-21 13:28:15 +01:00
# TODO:
# Following methods are forwarded directly to the web3 provider and therefore
# are not validated in any way.
lcProxy.proxy.registerProxyMethod("net_version")
lcProxy.proxy.registerProxyMethod("eth_call")
2022-09-28 22:43:10 +02:00
lcProxy.proxy.registerProxyMethod("eth_sendRawTransaction")
lcProxy.proxy.registerProxyMethod("eth_getTransactionReceipt")
# TODO currently we do not handle fullTransactions flag. It require updates on
# nim-web3 side
lcProxy.proxy.rpc("eth_getBlockByNumber") do(
quantityTag: BlockTag, fullTransactions: bool
) -> Opt[BlockObject]:
lcProxy.getBlockByTag(quantityTag)
lcProxy.proxy.rpc("eth_getBlockByHash") do(
blockHash: Hash32, fullTransactions: bool
) -> Opt[BlockObject]:
lcProxy.blockCache.getPayloadByHash(blockHash)
proc new*(
T: type VerifiedRpcProxy, proxy: RpcProxy, blockCache: BlockCache, chainId: Quantity
): T =
VerifiedRpcProxy(proxy: proxy, blockCache: blockCache, chainId: chainId)
# Used to be in eth1_monitor.nim; not sure why it was deleted,
# so I copied it here. --Adam
template awaitWithRetries*[T](
lazyFutExpr: Future[T], retries = 3, timeout = 60.seconds
): untyped =
const reqType = astToStr(lazyFutExpr)
var
retryDelayMs = 16000
f: Future[T]
attempts = 0
while true:
f = lazyFutExpr
yield f or sleepAsync(timeout)
if not f.finished:
await cancelAndWait(f)
elif f.failed:
when not (f.error of CatchableError):
static:
doAssert false, "f.error not CatchableError"
debug "Web3 request failed", req = reqType, err = f.error.msg
else:
break
inc attempts
if attempts >= retries:
var errorMsg = reqType & " failed " & $retries & " times"
if f.failed:
errorMsg &= ". Last error: " & f.error.msg
raise newException(ValueError, errorMsg)
await sleepAsync(chronos.milliseconds(retryDelayMs))
retryDelayMs *= 2
read(f)
proc verifyChaindId*(p: VerifiedRpcProxy): Future[void] {.async.} =
let localId = p.chainId
# retry 2 times, if the data provider fails despite the re-tries, propagate
# exception to the caller.
let providerId =
awaitWithRetries(p.rpcClient.eth_chainId(), retries = 2, timeout = seconds(30))
# This is a chain/network mismatch error between the Nimbus verified proxy and
# the application using it. Fail fast to avoid misusage. The user must fix
# the configuration.
if localId != providerId:
fatal "The specified data provider serves data for a different chain",
expectedChain = distinctBase(localId), providerChain = distinctBase(providerId)
quit 1
return