2022-11-10 09:29:34 +00:00
|
|
|
|
when (NimMajor, NimMinor) < (1, 4):
|
|
|
|
|
{.push raises: [Defect].}
|
|
|
|
|
else:
|
|
|
|
|
{.push raises: [].}
|
2022-12-05 19:02:21 +00:00
|
|
|
|
|
2022-11-10 09:29:34 +00:00
|
|
|
|
import
|
|
|
|
|
std/[json,tables,sequtils],
|
|
|
|
|
chronicles,
|
|
|
|
|
chronicles/topics_registry,
|
|
|
|
|
chronos,
|
|
|
|
|
json_serialization,
|
|
|
|
|
metrics,
|
|
|
|
|
metrics/chronos_httpserver,
|
|
|
|
|
presto/route,
|
|
|
|
|
presto/server,
|
|
|
|
|
stew/results,
|
|
|
|
|
stew/shims/net
|
|
|
|
|
|
|
|
|
|
logScope:
|
|
|
|
|
topics = "networkmonitor_metrics"
|
|
|
|
|
|
|
|
|
|
# On top of our custom metrics, the following are reused from nim-eth
|
|
|
|
|
#routing_table_nodes{state=""}
|
|
|
|
|
#routing_table_nodes{state="seen"}
|
|
|
|
|
#discovery_message_requests_outgoing_total{response=""}
|
|
|
|
|
#discovery_message_requests_outgoing_total{response="no_response"}
|
|
|
|
|
|
2023-09-28 08:07:27 +00:00
|
|
|
|
declarePublicGauge networkmonitor_peer_type_as_per_enr,
|
2022-11-10 09:29:34 +00:00
|
|
|
|
"Number of peers supporting each capability according the the ENR",
|
|
|
|
|
labels = ["capability"]
|
|
|
|
|
|
2023-09-28 08:07:27 +00:00
|
|
|
|
declarePublicGauge networkmonitor_peer_type_as_per_protocol,
|
2022-11-10 09:29:34 +00:00
|
|
|
|
"Number of peers supporting each protocol, after a successful connection) ",
|
|
|
|
|
labels = ["protocols"]
|
|
|
|
|
|
2023-09-28 08:07:27 +00:00
|
|
|
|
declarePublicGauge networkmonitor_peer_user_agents,
|
2022-11-10 09:29:34 +00:00
|
|
|
|
"Number of peers with each user agent",
|
|
|
|
|
labels = ["user_agent"]
|
|
|
|
|
|
2023-09-28 08:07:27 +00:00
|
|
|
|
declarePublicHistogram networkmonitor_peer_ping,
|
2023-09-25 12:38:59 +00:00
|
|
|
|
"Histogram tracking ping durations for discovered peers",
|
|
|
|
|
buckets = [100.0, 200.0, 300.0, 400.0, 500.0, 600.0, 700.0, 800.0, 900.0, 1000.0, 2000.0, Inf]
|
|
|
|
|
|
2023-09-28 08:07:27 +00:00
|
|
|
|
declarePublicGauge networkmonitor_peer_count,
|
|
|
|
|
"Number of discovered peers",
|
|
|
|
|
labels = ["connected"]
|
|
|
|
|
|
|
|
|
|
declarePublicGauge networkmonitor_peer_country_count,
|
|
|
|
|
"Number of peers per country",
|
|
|
|
|
labels = ["country"]
|
|
|
|
|
|
2022-11-10 09:29:34 +00:00
|
|
|
|
type
|
|
|
|
|
CustomPeerInfo* = object
|
|
|
|
|
# populated after discovery
|
2023-09-25 12:38:59 +00:00
|
|
|
|
lastTimeDiscovered*: int64
|
|
|
|
|
discovered*: int64
|
2022-11-10 09:29:34 +00:00
|
|
|
|
peerId*: string
|
|
|
|
|
enr*: string
|
|
|
|
|
ip*: string
|
|
|
|
|
enrCapabilities*: seq[string]
|
|
|
|
|
country*: string
|
|
|
|
|
city*: string
|
|
|
|
|
|
|
|
|
|
# only after ok connection
|
2023-09-25 12:38:59 +00:00
|
|
|
|
lastTimeConnected*: int64
|
|
|
|
|
retries*: int64
|
2022-11-10 09:29:34 +00:00
|
|
|
|
supportedProtocols*: seq[string]
|
|
|
|
|
userAgent*: string
|
2023-09-25 12:38:59 +00:00
|
|
|
|
lastPingDuration*: Duration
|
|
|
|
|
avgPingDuration*: Duration
|
2022-11-10 09:29:34 +00:00
|
|
|
|
|
2022-12-05 19:02:21 +00:00
|
|
|
|
# only after a ok/nok connection
|
|
|
|
|
connError*: string
|
|
|
|
|
|
2023-09-28 08:07:27 +00:00
|
|
|
|
CustomPeerInfoRef* = ref CustomPeerInfo
|
|
|
|
|
|
2022-11-14 07:33:36 +00:00
|
|
|
|
# Stores information about all discovered/connected peers
|
2023-09-28 08:07:27 +00:00
|
|
|
|
CustomPeersTableRef* = TableRef[string, CustomPeerInfoRef]
|
2022-11-10 09:29:34 +00:00
|
|
|
|
|
2022-11-14 07:33:36 +00:00
|
|
|
|
# stores the content topic and the count of rx messages
|
|
|
|
|
ContentTopicMessageTableRef* = TableRef[string, int]
|
|
|
|
|
|
|
|
|
|
proc installHandler*(router: var RestRouter,
|
|
|
|
|
allPeers: CustomPeersTableRef,
|
|
|
|
|
numMessagesPerContentTopic: ContentTopicMessageTableRef) =
|
2022-11-10 09:29:34 +00:00
|
|
|
|
router.api(MethodGet, "/allpeersinfo") do () -> RestApiResponse:
|
|
|
|
|
let values = toSeq(allPeers.values())
|
|
|
|
|
return RestApiResponse.response(values.toJson(), contentType="application/json")
|
2022-11-14 07:33:36 +00:00
|
|
|
|
router.api(MethodGet, "/contenttopics") do () -> RestApiResponse:
|
|
|
|
|
# TODO: toJson() includes the hash
|
|
|
|
|
return RestApiResponse.response($(%numMessagesPerContentTopic), contentType="application/json")
|
2022-11-10 09:29:34 +00:00
|
|
|
|
|
|
|
|
|
proc startMetricsServer*(serverIp: ValidIpAddress, serverPort: Port): Result[void, string] =
|
|
|
|
|
info "Starting metrics HTTP server", serverIp, serverPort
|
2022-12-05 19:02:21 +00:00
|
|
|
|
|
2022-11-10 09:29:34 +00:00
|
|
|
|
try:
|
|
|
|
|
startMetricsHttpServer($serverIp, serverPort)
|
|
|
|
|
except Exception as e:
|
|
|
|
|
error("Failed to start metrics HTTP server", serverIp=serverIp, serverPort=serverPort, msg=e.msg)
|
|
|
|
|
|
|
|
|
|
info "Metrics HTTP server started", serverIp, serverPort
|
2022-12-05 19:02:21 +00:00
|
|
|
|
ok()
|