mirror of
https://github.com/status-im/nim-eth-p2p.git
synced 2025-01-17 18:30:58 +00:00
465 lines
15 KiB
Nim
465 lines
15 KiB
Nim
#
|
|
# Ethereum P2P
|
|
# (c) Copyright 2018
|
|
# Status Research & Development GmbH
|
|
#
|
|
# Licensed under either of
|
|
# Apache License, version 2.0, (LICENSE-APACHEv2)
|
|
# MIT license (LICENSE-MIT)
|
|
#
|
|
|
|
import
|
|
times, tables, options, sets, hashes, strutils, macros,
|
|
chronicles, asyncdispatch2, nimcrypto/[keccak, hash],
|
|
rlp, eth_common/eth_types, eth_keys,
|
|
../rlpx, ../kademlia, ../private/types, ../blockchain_utils,
|
|
les/private/les_types, les/flow_control
|
|
|
|
les_types.forwardPublicTypes
|
|
|
|
const
|
|
lesVersion = 2'u
|
|
maxHeadersFetch = 192
|
|
maxBodiesFetch = 32
|
|
maxReceiptsFetch = 128
|
|
maxCodeFetch = 64
|
|
maxProofsFetch = 64
|
|
maxHeaderProofsFetch = 64
|
|
maxTransactionsFetch = 64
|
|
|
|
# Handshake properties:
|
|
# https://github.com/zsfelfoldi/go-ethereum/wiki/Light-Ethereum-Subprotocol-(LES)
|
|
keyProtocolVersion = "protocolVersion"
|
|
## P: is 1 for the LPV1 protocol version.
|
|
|
|
keyNetworkId = "networkId"
|
|
## P: should be 0 for testnet, 1 for mainnet.
|
|
|
|
keyHeadTotalDifficulty = "headTd"
|
|
## P: Total Difficulty of the best chain.
|
|
## Integer, as found in block header.
|
|
|
|
keyHeadHash = "headHash"
|
|
## B_32: the hash of the best (i.e. highest TD) known block.
|
|
|
|
keyHeadNumber = "headNum"
|
|
## P: the number of the best (i.e. highest TD) known block.
|
|
|
|
keyGenesisHash = "genesisHash"
|
|
## B_32: the hash of the Genesis block.
|
|
|
|
keyServeHeaders = "serveHeaders"
|
|
## (optional, no value)
|
|
## present if the peer can serve header chain downloads.
|
|
|
|
keyServeChainSince = "serveChainSince"
|
|
## P (optional)
|
|
## present if the peer can serve Body/Receipts ODR requests
|
|
## starting from the given block number.
|
|
|
|
keyServeStateSince = "serveStateSince"
|
|
## P (optional):
|
|
## present if the peer can serve Proof/Code ODR requests
|
|
## starting from the given block number.
|
|
|
|
keyRelaysTransactions = "txRelay"
|
|
## (optional, no value)
|
|
## present if the peer can relay transactions to the ETH network.
|
|
|
|
keyFlowControlBL = "flowControl/BL"
|
|
keyFlowControlMRC = "flowControl/MRC"
|
|
keyFlowControlMRR = "flowControl/MRR"
|
|
## see Client Side Flow Control:
|
|
## https://github.com/zsfelfoldi/go-ethereum/wiki/Client-Side-Flow-Control-model-for-the-LES-protocol
|
|
|
|
keyAnnounceType = "announceType"
|
|
keyAnnounceSignature = "sign"
|
|
|
|
proc initProtocolState(network: LesNetwork, node: EthereumNode) =
|
|
network.peers = initSet[LesPeer]()
|
|
|
|
proc addPeer(network: LesNetwork, peer: LesPeer) =
|
|
network.enlistInFlowControl peer
|
|
network.peers.incl peer
|
|
|
|
proc removePeer(network: LesNetwork, peer: LesPeer) =
|
|
network.delistFromFlowControl peer
|
|
network.peers.excl peer
|
|
|
|
template costQuantity(quantityExpr, max: untyped) {.pragma.}
|
|
|
|
proc getCostQuantity(fn: NimNode): tuple[quantityExpr, maxQuantity: NimNode] =
|
|
# XXX: `getCustomPragmaVal` doesn't work yet on regular nnkProcDef nodes
|
|
# (TODO: file as an issue)
|
|
let p = fn.pragma
|
|
assert p.kind == nnkPragma and p.len > 0 and $p[0][0] == "costQuantity"
|
|
|
|
result.quantityExpr = p[0][1]
|
|
result.maxQuantity= p[0][2]
|
|
|
|
if result.maxQuantity.kind == nnkExprEqExpr:
|
|
result.maxQuantity = result.maxQuantity[1]
|
|
|
|
macro outgoingRequestDecorator(n: untyped): untyped =
|
|
result = n
|
|
let (costQuantity, maxQuantity) = n.getCostQuantity
|
|
|
|
result.body.add quote do:
|
|
trackOutgoingRequest(msgRecipient.networkState(les),
|
|
msgRecipient.state(les),
|
|
perProtocolMsgId, reqId, `costQuantity`)
|
|
# echo result.repr
|
|
|
|
macro incomingResponseDecorator(n: untyped): untyped =
|
|
result = n
|
|
|
|
let trackingCall = quote do:
|
|
trackIncomingResponse(msgSender.state(les), reqId, msg.bufValue)
|
|
|
|
result.body.insert(n.body.len - 1, trackingCall)
|
|
# echo result.repr
|
|
|
|
macro incomingRequestDecorator(n: untyped): untyped =
|
|
result = n
|
|
let (costQuantity, maxQuantity) = n.getCostQuantity
|
|
|
|
template acceptStep(quantityExpr, maxQuantity) {.dirty.} =
|
|
let requestCostQuantity = quantityExpr
|
|
if requestCostQuantity > maxQuantity:
|
|
await peer.disconnect(BreachOfProtocol)
|
|
return
|
|
|
|
let lesPeer = peer.state
|
|
let lesNetwork = peer.networkState
|
|
|
|
if not await acceptRequest(lesNetwork, lesPeer,
|
|
perProtocolMsgId,
|
|
requestCostQuantity): return
|
|
|
|
result.body.insert(1, getAst(acceptStep(costQuantity, maxQuantity)))
|
|
# echo result.repr
|
|
|
|
template updateBV: BufValueInt =
|
|
bufValueAfterRequest(lesNetwork, lesPeer,
|
|
perProtocolMsgId, requestCostQuantity)
|
|
|
|
func getValue(values: openarray[KeyValuePair],
|
|
key: string, T: typedesc): Option[T] =
|
|
for v in values:
|
|
if v.key == key:
|
|
return some(rlp.decode(v.value, T))
|
|
|
|
func getRequiredValue(values: openarray[KeyValuePair],
|
|
key: string, T: typedesc): T =
|
|
for v in values:
|
|
if v.key == key:
|
|
return rlp.decode(v.value, T)
|
|
|
|
raise newException(HandshakeError,
|
|
"Required handshake field " & key & " missing")
|
|
|
|
p2pProtocol les(version = lesVersion,
|
|
peerState = LesPeer,
|
|
networkState = LesNetwork,
|
|
outgoingRequestDecorator = outgoingRequestDecorator,
|
|
incomingRequestDecorator = incomingRequestDecorator,
|
|
incomingResponseThunkDecorator = incomingResponseDecorator):
|
|
|
|
## Handshake
|
|
##
|
|
|
|
proc status(p: Peer, values: openarray[KeyValuePair])
|
|
|
|
onPeerConnected do (peer: Peer):
|
|
let
|
|
network = peer.network
|
|
chain = network.chain
|
|
bestBlock = chain.getBestBlockHeader
|
|
lesPeer = peer.state
|
|
lesNetwork = peer.networkState
|
|
|
|
template `=>`(k, v: untyped): untyped =
|
|
KeyValuePair.init(key = k, value = rlp.encode(v))
|
|
|
|
var lesProperties = @[
|
|
keyProtocolVersion => lesVersion,
|
|
keyNetworkId => network.networkId,
|
|
keyHeadTotalDifficulty => bestBlock.difficulty,
|
|
keyHeadHash => bestBlock.blockHash,
|
|
keyHeadNumber => bestBlock.blockNumber,
|
|
keyGenesisHash => chain.genesisHash
|
|
]
|
|
|
|
lesPeer.remoteReqCosts = currentRequestsCosts(lesNetwork, les.protocolInfo)
|
|
|
|
if lesNetwork.areWeServingData:
|
|
lesProperties.add [
|
|
# keyServeHeaders => nil,
|
|
keyServeChainSince => 0,
|
|
keyServeStateSince => 0,
|
|
# keyRelaysTransactions => nil,
|
|
keyFlowControlBL => lesNetwork.bufferLimit,
|
|
keyFlowControlMRR => lesNetwork.minRechargingRate,
|
|
keyFlowControlMRC => lesPeer.remoteReqCosts
|
|
]
|
|
|
|
if lesNetwork.areWeRequestingData:
|
|
lesProperties.add(keyAnnounceType => lesNetwork.ourAnnounceType)
|
|
|
|
let
|
|
s = await peer.nextMsg(les.status)
|
|
peerNetworkId = s.values.getRequiredValue(keyNetworkId, uint)
|
|
peerGenesisHash = s.values.getRequiredValue(keyGenesisHash, KeccakHash)
|
|
peerLesVersion = s.values.getRequiredValue(keyProtocolVersion, uint)
|
|
|
|
template requireCompatibility(peerVar, localVar, varName: untyped) =
|
|
if localVar != peerVar:
|
|
raise newException(HandshakeError,
|
|
"Incompatibility detected! $1 mismatch ($2 != $3)" %
|
|
[varName, $localVar, $peerVar])
|
|
|
|
requireCompatibility(peerLesVersion, lesVersion, "les version")
|
|
requireCompatibility(peerNetworkId, network.networkId, "network id")
|
|
requireCompatibility(peerGenesisHash, chain.genesisHash, "genesis hash")
|
|
|
|
template `:=`(lhs, key) =
|
|
lhs = s.values.getRequiredValue(key, type(lhs))
|
|
|
|
lesPeer.bestBlockHash := keyHeadHash
|
|
lesPeer.bestBlockNumber := keyHeadNumber
|
|
lesPeer.bestDifficulty := keyHeadTotalDifficulty
|
|
|
|
let peerAnnounceType = s.values.getValue(keyAnnounceType, AnnounceType)
|
|
if peerAnnounceType.isSome:
|
|
lesPeer.isClient = true
|
|
lesPeer.announceType = peerAnnounceType.get
|
|
else:
|
|
lesPeer.announceType = AnnounceType.Simple
|
|
lesPeer.hasChainSince := keyServeChainSince
|
|
lesPeer.hasStateSince := keyServeStateSince
|
|
lesPeer.relaysTransactions := keyRelaysTransactions
|
|
lesPeer.localFlowState.bufLimit := keyFlowControlBL
|
|
lesPeer.localFlowState.minRecharge := keyFlowControlMRR
|
|
lesPeer.localReqCosts := keyFlowControlMRC
|
|
|
|
lesNetwork.addPeer lesPeer
|
|
|
|
onPeerDisconnected do (peer: Peer, reason: DisconnectionReason) {.gcsafe.}:
|
|
peer.networkState.removePeer peer.state
|
|
|
|
## Header synchronisation
|
|
##
|
|
|
|
proc announce(
|
|
peer: Peer,
|
|
headHash: KeccakHash,
|
|
headNumber: BlockNumber,
|
|
headTotalDifficulty: DifficultyInt,
|
|
reorgDepth: BlockNumber,
|
|
values: openarray[KeyValuePair],
|
|
announceType: AnnounceType) =
|
|
|
|
if peer.state.announceType == AnnounceType.None:
|
|
error "unexpected announce message", peer
|
|
return
|
|
|
|
if announceType == AnnounceType.Signed:
|
|
let signature = values.getValue(keyAnnounceSignature, Blob)
|
|
if signature.isNone:
|
|
error "missing announce signature"
|
|
return
|
|
let sigHash = keccak256.digest rlp.encodeList(headHash,
|
|
headNumber,
|
|
headTotalDifficulty)
|
|
let signerKey = recoverKeyFromSignature(signature.get.initSignature,
|
|
sigHash)
|
|
if signerKey.toNodeId != peer.remote.id:
|
|
error "invalid announce signature"
|
|
# TODO: should we disconnect this peer?
|
|
return
|
|
|
|
# TODO: handle new block
|
|
|
|
requestResponse:
|
|
proc getBlockHeaders(
|
|
peer: Peer,
|
|
req: BlocksRequest) {.
|
|
costQuantity(req.maxResults.int, max = maxHeadersFetch).} =
|
|
|
|
let headers = peer.network.chain.getBlockHeaders(req)
|
|
await peer.blockHeaders(reqId, updateBV(), headers)
|
|
|
|
proc blockHeaders(
|
|
peer: Peer,
|
|
bufValue: BufValueInt,
|
|
blocks: openarray[BlockHeader])
|
|
|
|
## On-damand data retrieval
|
|
##
|
|
|
|
requestResponse:
|
|
proc getBlockBodies(
|
|
peer: Peer,
|
|
blocks: openarray[KeccakHash]) {.
|
|
costQuantity(blocks.len, max = maxBodiesFetch).} =
|
|
|
|
let blocks = peer.network.chain.getBlockBodies(blocks)
|
|
await peer.blockBodies(reqId, updateBV(), blocks)
|
|
|
|
proc blockBodies(
|
|
peer: Peer,
|
|
bufValue: BufValueInt,
|
|
bodies: openarray[BlockBody])
|
|
|
|
requestResponse:
|
|
proc getReceipts(
|
|
peer: Peer,
|
|
hashes: openarray[KeccakHash])
|
|
{.costQuantity(hashes.len, max = maxReceiptsFetch).} =
|
|
|
|
let receipts = peer.network.chain.getReceipts(hashes)
|
|
await peer.receipts(reqId, updateBV(), receipts)
|
|
|
|
proc receipts(
|
|
peer: Peer,
|
|
bufValue: BufValueInt,
|
|
receipts: openarray[Receipt])
|
|
|
|
requestResponse:
|
|
proc getProofs(
|
|
peer: Peer,
|
|
proofs: openarray[ProofRequest]) {.
|
|
costQuantity(proofs.len, max = maxProofsFetch).} =
|
|
|
|
let proofs = peer.network.chain.getProofs(proofs)
|
|
await peer.proofs(reqId, updateBV(), proofs)
|
|
|
|
proc proofs(
|
|
peer: Peer,
|
|
bufValue: BufValueInt,
|
|
proofs: openarray[Blob])
|
|
|
|
requestResponse:
|
|
proc getContractCodes(
|
|
peer: Peer,
|
|
reqs: seq[ContractCodeRequest]) {.
|
|
costQuantity(reqs.len, max = maxCodeFetch).} =
|
|
|
|
let results = peer.network.chain.getContractCodes(reqs)
|
|
await peer.contractCodes(reqId, updateBV(), results)
|
|
|
|
proc contractCodes(
|
|
peer: Peer,
|
|
bufValue: BufValueInt,
|
|
results: seq[Blob])
|
|
|
|
nextID 15
|
|
|
|
requestResponse:
|
|
proc getHeaderProofs(
|
|
peer: Peer,
|
|
reqs: openarray[ProofRequest]) {.
|
|
costQuantity(reqs.len, max = maxHeaderProofsFetch).} =
|
|
|
|
let proofs = peer.network.chain.getHeaderProofs(reqs)
|
|
await peer.headerProofs(reqId, updateBV(), proofs)
|
|
|
|
proc headerProofs(
|
|
peer: Peer,
|
|
bufValue: BufValueInt,
|
|
proofs: openarray[Blob])
|
|
|
|
requestResponse:
|
|
proc getHelperTrieProofs(
|
|
peer: Peer,
|
|
reqs: openarray[HelperTrieProofRequest]) {.
|
|
costQuantity(reqs.len, max = maxProofsFetch).} =
|
|
|
|
var nodes, auxData: seq[Blob]
|
|
peer.network.chain.getHelperTrieProofs(reqs, nodes, auxData)
|
|
await peer.helperTrieProofs(reqId, updateBV(), nodes, auxData)
|
|
|
|
proc helperTrieProofs(
|
|
peer: Peer,
|
|
bufValue: BufValueInt,
|
|
nodes: seq[Blob],
|
|
auxData: seq[Blob])
|
|
|
|
## Transaction relaying and status retrieval
|
|
##
|
|
|
|
requestResponse:
|
|
proc sendTxV2(
|
|
peer: Peer,
|
|
transactions: openarray[Transaction]) {.
|
|
costQuantity(transactions.len, max = maxTransactionsFetch).} =
|
|
|
|
let chain = peer.network.chain
|
|
|
|
var results: seq[TransactionStatusMsg]
|
|
for t in transactions:
|
|
let hash = t.rlpHash # TODO: this is not optimal, we can compute
|
|
# the hash from the request bytes.
|
|
# The RLP module can offer a helper Hashed[T]
|
|
# to make this easy.
|
|
var s = chain.getTransactionStatus(hash)
|
|
if s.status == TransactionStatus.Unknown:
|
|
chain.addTransactions([t])
|
|
s = chain.getTransactionStatus(hash)
|
|
|
|
results.add s
|
|
|
|
await peer.txStatus(reqId, updateBV(), results)
|
|
|
|
proc getTxStatus(
|
|
peer: Peer,
|
|
transactions: openarray[Transaction]) {.
|
|
costQuantity(transactions.len, max = maxTransactionsFetch).} =
|
|
|
|
let chain = peer.network.chain
|
|
|
|
var results: seq[TransactionStatusMsg]
|
|
for t in transactions:
|
|
results.add chain.getTransactionStatus(t.rlpHash)
|
|
await peer.txStatus(reqId, updateBV(), results)
|
|
|
|
proc txStatus(
|
|
peer: Peer,
|
|
bufValue: BufValueInt,
|
|
transactions: openarray[TransactionStatusMsg])
|
|
|
|
proc configureLes*(node: EthereumNode,
|
|
# Client options:
|
|
announceType = AnnounceType.Simple,
|
|
# Server options.
|
|
# The zero default values indicate that the
|
|
# LES server will be deactivated.
|
|
maxReqCount = 0,
|
|
maxReqCostSum = 0,
|
|
reqCostTarget = 0) =
|
|
|
|
doAssert announceType != AnnounceType.Unspecified or maxReqCount > 0
|
|
|
|
var lesNetwork = node.protocolState(les)
|
|
lesNetwork.ourAnnounceType = announceType
|
|
initFlowControl(lesNetwork, les.protocolInfo,
|
|
maxReqCount, maxReqCostSum, reqCostTarget,
|
|
node.chain)
|
|
|
|
proc configureLesServer*(node: EthereumNode,
|
|
# Client options:
|
|
announceType = AnnounceType.Unspecified,
|
|
# Server options.
|
|
# The zero default values indicate that the
|
|
# LES server will be deactivated.
|
|
maxReqCount = 0,
|
|
maxReqCostSum = 0,
|
|
reqCostTarget = 0) =
|
|
## This is similar to `configureLes`, but with default parameter
|
|
## values appropriate for a server.
|
|
node.configureLes(announceType, maxReqCount, maxReqCostSum, reqCostTarget)
|
|
|
|
proc persistLesMessageStats*(node: EthereumNode) =
|
|
persistMessageStats(node.chain, node.protocolState(les))
|
|
|