# Copyright (c) 2021-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 stew/byteutils, chronicles, eth/p2p/discoveryv5/enr, libp2p/[multiaddress, multicodec, peerstore], ../version, ../beacon_node, ../sync/sync_manager, ../networking/[eth2_network, peer_pool], ../spec/datatypes/base, ./rest_utils export rest_utils logScope: topics = "rest_node" type ConnectionStateSet* = set[ConnectionState] PeerTypeSet* = set[PeerType] RestNodePeerCount* = object disconnected*: uint64 connecting*: uint64 connected*: uint64 disconnecting*: uint64 RestJson.useDefaultSerializationFor( RestNodePeerCount, ) proc normalize*(address: MultiAddress, value: PeerId): MaResult[MultiAddress] = ## Checks if `address` has `p2p` suffix, and if not add it. let protos = ? address.protocols() index = protos.find(multiCodec("p2p")) if index == -1: let suffix = ? MultiAddress.init(multiCodec("p2p"), value) concat(address, suffix) else: ok(address) proc validateState(states: seq[PeerStateKind]): Result[ConnectionStateSet, cstring] = var res: set[ConnectionState] for item in states: case item of PeerStateKind.Disconnected: if ConnectionState.Disconnected in res: return err("Peer connection states must be unique") res.incl(ConnectionState.Disconnected) of PeerStateKind.Connecting: if ConnectionState.Connecting in res: return err("Peer connection states must be unique") res.incl(ConnectionState.Connecting) of PeerStateKind.Connected: if ConnectionState.Connected in res: return err("Peer connection states must be unique") res.incl(ConnectionState.Connected) of PeerStateKind.Disconnecting: if ConnectionState.Disconnecting in res: return err("Peer connection states must be unique") res.incl(ConnectionState.Disconnecting) if res == {}: res = {ConnectionState.Connecting, ConnectionState.Connected, ConnectionState.Disconnecting, ConnectionState.Disconnected} ok(res) proc validateDirection(directions: seq[PeerDirectKind]): Result[PeerTypeSet, cstring] = var res: set[PeerType] for item in directions: case item of PeerDirectKind.Inbound: if PeerType.Incoming in res: return err("Peer direction states must be unique") res.incl(PeerType.Incoming) of PeerDirectKind.Outbound: if PeerType.Outgoing in res: return err("Peer direction states must be unique") res.incl(PeerType.Outgoing) if res == {}: res = {PeerType.Incoming, PeerType.Outgoing} ok(res) proc toString(state: ConnectionState): string = case state of ConnectionState.Disconnected: "disconnected" of ConnectionState.Connecting: "connecting" of ConnectionState.Connected: "connected" of ConnectionState.Disconnecting: "disconnecting" else: "" proc toString(direction: PeerType): string = case direction: of PeerType.Incoming: "inbound" of PeerType.Outgoing: "outbound" proc getLastSeenAddress(node: BeaconNode, id: PeerId): string = let address = node.network.switch.peerStore[LastSeenBook][id].valueOr: return "" normalized = address.normalize(id).valueOr: return "" $normalized proc getDiscoveryAddresses(node: BeaconNode): seq[string] = let typedRec = TypedRecord.fromRecord(node.network.enrRecord()) peerAddr = typedRec.toPeerAddr(udpProtocol).valueOr: return default(seq[string]) maddress = MultiAddress.init(multiCodec("p2p"), peerAddr.peerId).valueOr: return default(seq[string]) var addresses: seq[string] for item in peerAddr.addrs: let res = concat(item, maddress) if res.isOk(): addresses.add($(res.get())) addresses proc getP2PAddresses(node: BeaconNode): seq[string] = let pinfo = node.network.switch.peerInfo maddress = MultiAddress.init(multiCodec("p2p"), pinfo.peerId).valueOr: return default(seq[string]) var addresses: seq[string] for item in node.network.announcedAddresses: let res = concat(item, maddress) if res.isOk(): addresses.add($(res.get())) for item in pinfo.addrs: let res = concat(item, maddress) if res.isOk(): addresses.add($(res.get())) addresses proc installNodeApiHandlers*(router: var RestRouter, node: BeaconNode) = let cachedVersion = RestApiResponse.prepareJsonResponse((version: nimbusAgentStr)) # https://ethereum.github.io/beacon-APIs/#/Node/getNetworkIdentity router.api2(MethodGet, "/eth/v1/node/identity") do () -> RestApiResponse: RestApiResponse.jsonResponse( ( peer_id: $node.network.peerId(), enr: node.network.enrRecord().toURI(), p2p_addresses: node.getP2PAddresses(), discovery_addresses: node.getDiscoveryAddresses(), metadata: ( seq_number: node.network.metadata.seq_number, syncnets: to0xHex(node.network.metadata.syncnets.bytes), attnets: to0xHex(node.network.metadata.attnets.bytes) ) ) ) # https://ethereum.github.io/beacon-APIs/#/Node/getPeers router.api2(MethodGet, "/eth/v1/node/peers") do ( state: seq[PeerStateKind], direction: seq[PeerDirectKind]) -> RestApiResponse: let connectionMask = block: if state.isErr(): return RestApiResponse.jsonError(Http400, InvalidPeerStateValueError, $state.error()) validateState(state.get()).valueOr: return RestApiResponse.jsonError(Http400, InvalidPeerStateValueError, $error) let directionMask = block: if direction.isErr(): return RestApiResponse.jsonError(Http400, InvalidPeerDirectionValueError, $direction.error()) validateDirection(direction.get()).valueOr: return RestApiResponse.jsonError(Http400, InvalidPeerDirectionValueError, $error) var res: seq[RestNodePeer] for peer in node.network.peers.values(): if (peer.connectionState in connectionMask) and (peer.direction in directionMask): let peer = RestNodePeer( peer_id: $peer.peerId, enr: if peer.enr.isSome(): peer.enr.get().toURI() else: "", last_seen_p2p_address: getLastSeenAddress(node, peer.peerId), state: peer.connectionState.toString(), direction: peer.direction.toString(), # Fields `agent` and `proto` are not part of specification agent: node.network.switch.peerStore[AgentBook][peer.peerId], proto: node.network.switch.peerStore[ProtoVersionBook][peer.peerId] ) res.add(peer) RestApiResponse.jsonResponseWMeta(res, (count: RestNumeric(len(res)))) # https://ethereum.github.io/beacon-APIs/#/Node/getPeerCount router.api2(MethodGet, "/eth/v1/node/peer_count") do () -> RestApiResponse: var res: RestNodePeerCount for item in node.network.peers.values(): case item.connectionState of ConnectionState.Connecting: inc(res.connecting) of ConnectionState.Connected: inc(res.connected) of ConnectionState.Disconnecting: inc(res.disconnecting) of ConnectionState.Disconnected: inc(res.disconnected) of ConnectionState.None: discard RestApiResponse.jsonResponse(res) # https://ethereum.github.io/beacon-APIs/#/Node/getPeer router.api2(MethodGet, "/eth/v1/node/peers/{peer_id}") do ( peer_id: PeerId) -> RestApiResponse: let peer = block: if peer_id.isErr(): return RestApiResponse.jsonError(Http400, InvalidPeerIdValueError, $peer_id.error()) let res = node.network.peers.getOrDefault(peer_id.get()) if isNil(res): return RestApiResponse.jsonError(Http404, PeerNotFoundError) res RestApiResponse.jsonResponse( ( peer_id: $peer.peerId, enr: if peer.enr.isSome(): peer.enr.get().toURI() else: "", last_seen_p2p_address: getLastSeenAddress(node, peer.peerId), state: peer.connectionState.toString(), direction: peer.direction.toString(), agent: node.network.switch.peerStore[AgentBook][peer.peerId], # Fields `agent` and `proto` are not part of specification proto: node.network.switch.peerStore[ProtoVersionBook][peer.peerId] # Fields `agent` and `proto` are not part of specification ) ) # https://ethereum.github.io/beacon-APIs/#/Node/getNodeVersion router.api2(MethodGet, "/eth/v1/node/version") do () -> RestApiResponse: RestApiResponse.response(cachedVersion, Http200, "application/json") # https://ethereum.github.io/beacon-APIs/#/Node/getSyncingStatus router.api2(MethodGet, "/eth/v1/node/syncing") do () -> RestApiResponse: let wallSlot = node.beaconClock.now().slotOrZero() headSlot = node.dag.head.slot distance = wallSlot - headSlot isSyncing = if isNil(node.syncManager): false else: node.syncManager.inProgress isOptimistic = if node.currentSlot().epoch() >= node.dag.cfg.BELLATRIX_FORK_EPOCH: some(not node.dag.head.executionValid) else: none[bool]() elOffline = if node.currentSlot().epoch() >= node.dag.cfg.CAPELLA_FORK_EPOCH: some(not node.elManager.hasAnyWorkingConnection) else: none[bool]() # Added with ethereum/beacon-APIs v2.4.0 info = RestSyncInfo( head_slot: headSlot, sync_distance: distance, is_syncing: isSyncing, is_optimistic: isOptimistic, el_offline: elOffline ) RestApiResponse.jsonResponse(info) # https://ethereum.github.io/beacon-APIs/#/Node/getHealth router.api2(MethodGet, "/eth/v1/node/health") do () -> RestApiResponse: # TODO: Add ability to detect node's issues and return 503 error according # to specification. let status = if node.syncManager.inProgress: Http206 else: Http200 RestApiResponse.response(status)