Browse Source

p2p: add message type into the send/recv bytes metrics (backport #7155) (#7161)

* p2p: add message type into the send/recv bytes metrics (#7155)

This pull request adds a new "mesage_type" label to the send/recv bytes metrics calculated in the p2p code.

Below is a snippet of the updated metrics that includes the updated label:
```
tendermint_p2p_peer_receive_bytes_total{chID="32",chain_id="ci",message_type="consensus_HasVote",peer_id="2551a13ed720101b271a5df4816d1e4b3d3bd133"} 652
tendermint_p2p_peer_receive_bytes_total{chID="32",chain_id="ci",message_type="consensus_HasVote",peer_id="4b1068420ef739db63377250553562b9a978708a"} 631
tendermint_p2p_peer_receive_bytes_total{chID="32",chain_id="ci",message_type="consensus_HasVote",peer_id="927c50a5e508c747830ce3ba64a3f70fdda58ef2"} 631
tendermint_p2p_peer_receive_bytes_total{chID="32",chain_id="ci",message_type="consensus_NewRoundStep",peer_id="2551a13ed720101b271a5df4816d1e4b3d3bd133"} 393
tendermint_p2p_peer_receive_bytes_total{chID="32",chain_id="ci",message_type="consensus_NewRoundStep",peer_id="4b1068420ef739db63377250553562b9a978708a"} 357
tendermint_p2p_peer_receive_bytes_total{chID="32",chain_id="ci",message_type="consensus_NewRoundStep",peer_id="927c50a5e508c747830ce3ba64a3f70fdda58ef2"} 386
```

(cherry picked from commit b4bc6bb4e8)
pull/7179/head
mergify[bot] 3 years ago
committed by GitHub
parent
commit
dd1471da91
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 68 additions and 4 deletions
  1. +45
    -2
      internal/p2p/metrics.go
  2. +19
    -0
      internal/p2p/metrics_test.go
  3. +2
    -1
      internal/p2p/pqueue.go
  4. +2
    -1
      internal/p2p/router.go

+ 45
- 2
internal/p2p/metrics.go View File

@ -1,6 +1,11 @@
package p2p package p2p
import ( import (
"fmt"
"reflect"
"regexp"
"sync"
"github.com/go-kit/kit/metrics" "github.com/go-kit/kit/metrics"
"github.com/go-kit/kit/metrics/discard" "github.com/go-kit/kit/metrics/discard"
"github.com/go-kit/kit/metrics/prometheus" "github.com/go-kit/kit/metrics/prometheus"
@ -13,6 +18,13 @@ const (
MetricsSubsystem = "p2p" MetricsSubsystem = "p2p"
) )
var (
// valueToLabelRegexp is used to find the golang package name and type name
// so that the name can be turned into a prometheus label where the characters
// in the label do not include prometheus special characters such as '*' and '.'.
valueToLabelRegexp = regexp.MustCompile(`\*?(\w+)\.(.*)`)
)
// Metrics contains metrics exposed by this package. // Metrics contains metrics exposed by this package.
type Metrics struct { type Metrics struct {
// Number of peers. // Number of peers.
@ -43,6 +55,9 @@ type Metrics struct {
// PeerQueueMsgSize defines the average size of messages sent over a peer's // PeerQueueMsgSize defines the average size of messages sent over a peer's
// queue for a specific flow (i.e. Channel). // queue for a specific flow (i.e. Channel).
PeerQueueMsgSize metrics.Gauge PeerQueueMsgSize metrics.Gauge
mtx *sync.RWMutex
messageLabelNames map[reflect.Type]string
} }
// PrometheusMetrics returns Metrics build using Prometheus client library. // PrometheusMetrics returns Metrics build using Prometheus client library.
@ -68,14 +83,14 @@ func PrometheusMetrics(namespace string, labelsAndValues ...string) *Metrics {
Subsystem: MetricsSubsystem, Subsystem: MetricsSubsystem,
Name: "peer_receive_bytes_total", Name: "peer_receive_bytes_total",
Help: "Number of bytes received from a given peer.", Help: "Number of bytes received from a given peer.",
}, append(labels, "peer_id", "chID")).With(labelsAndValues...),
}, append(labels, "peer_id", "chID", "message_type")).With(labelsAndValues...),
PeerSendBytesTotal: prometheus.NewCounterFrom(stdprometheus.CounterOpts{ PeerSendBytesTotal: prometheus.NewCounterFrom(stdprometheus.CounterOpts{
Namespace: namespace, Namespace: namespace,
Subsystem: MetricsSubsystem, Subsystem: MetricsSubsystem,
Name: "peer_send_bytes_total", Name: "peer_send_bytes_total",
Help: "Number of bytes sent to a given peer.", Help: "Number of bytes sent to a given peer.",
}, append(labels, "peer_id", "chID")).With(labelsAndValues...),
}, append(labels, "peer_id", "chID", "message_type")).With(labelsAndValues...),
PeerPendingSendBytes: prometheus.NewGaugeFrom(stdprometheus.GaugeOpts{ PeerPendingSendBytes: prometheus.NewGaugeFrom(stdprometheus.GaugeOpts{
Namespace: namespace, Namespace: namespace,
@ -118,6 +133,9 @@ func PrometheusMetrics(namespace string, labelsAndValues ...string) *Metrics {
Name: "router_channel_queue_msg_size", Name: "router_channel_queue_msg_size",
Help: "The size of messages sent over a peer's queue for a specific p2p Channel.", Help: "The size of messages sent over a peer's queue for a specific p2p Channel.",
}, append(labels, "ch_id")).With(labelsAndValues...), }, append(labels, "ch_id")).With(labelsAndValues...),
mtx: &sync.RWMutex{},
messageLabelNames: map[reflect.Type]string{},
} }
} }
@ -133,5 +151,30 @@ func NopMetrics() *Metrics {
RouterChannelQueueSend: discard.NewHistogram(), RouterChannelQueueSend: discard.NewHistogram(),
PeerQueueDroppedMsgs: discard.NewCounter(), PeerQueueDroppedMsgs: discard.NewCounter(),
PeerQueueMsgSize: discard.NewGauge(), PeerQueueMsgSize: discard.NewGauge(),
mtx: &sync.RWMutex{},
messageLabelNames: map[reflect.Type]string{},
}
}
// ValueToMetricLabel is a method that is used to produce a prometheus label value of the golang
// type that is passed in.
// This method uses a map on the Metrics struct so that each label name only needs
// to be produced once to prevent expensive string operations.
func (m *Metrics) ValueToMetricLabel(i interface{}) string {
t := reflect.TypeOf(i)
m.mtx.RLock()
if s, ok := m.messageLabelNames[t]; ok {
m.mtx.RUnlock()
return s
} }
m.mtx.RUnlock()
s := t.String()
ss := valueToLabelRegexp.FindStringSubmatch(s)
l := fmt.Sprintf("%s_%s", ss[1], ss[2])
m.mtx.Lock()
defer m.mtx.Unlock()
m.messageLabelNames[t] = l
return l
} }

+ 19
- 0
internal/p2p/metrics_test.go View File

@ -0,0 +1,19 @@
package p2p
import (
"testing"
"github.com/stretchr/testify/assert"
"github.com/tendermint/tendermint/proto/tendermint/p2p"
)
func TestValueToMetricsLabel(t *testing.T) {
m := NopMetrics()
r := &p2p.PexResponse{}
str := m.ValueToMetricLabel(r)
assert.Equal(t, "p2p_PexResponse", str)
// subsequent calls to the function should produce the same result
str = m.ValueToMetricLabel(r)
assert.Equal(t, "p2p_PexResponse", str)
}

+ 2
- 1
internal/p2p/pqueue.go View File

@ -256,7 +256,8 @@ func (s *pqScheduler) process() {
s.metrics.PeerSendBytesTotal.With( s.metrics.PeerSendBytesTotal.With(
"chID", chIDStr, "chID", chIDStr,
"peer_id", string(pqEnv.envelope.To)).Add(float64(pqEnv.size))
"peer_id", string(pqEnv.envelope.To),
"message_type", s.metrics.ValueToMetricLabel(pqEnv.envelope.Message)).Add(float64(pqEnv.size))
select { select {
case s.dequeueCh <- pqEnv.envelope: case s.dequeueCh <- pqEnv.envelope:
case <-s.closer.Done(): case <-s.closer.Done():


+ 2
- 1
internal/p2p/router.go View File

@ -938,7 +938,8 @@ func (r *Router) receivePeer(peerID types.NodeID, conn Connection) error {
case queue.enqueue() <- Envelope{From: peerID, Message: msg}: case queue.enqueue() <- Envelope{From: peerID, Message: msg}:
r.metrics.PeerReceiveBytesTotal.With( r.metrics.PeerReceiveBytesTotal.With(
"chID", fmt.Sprint(chID), "chID", fmt.Sprint(chID),
"peer_id", string(peerID)).Add(float64(proto.Size(msg)))
"peer_id", string(peerID),
"message_type", r.metrics.ValueToMetricLabel(msg)).Add(float64(proto.Size(msg)))
r.metrics.RouterChannelQueueSend.Observe(time.Since(start).Seconds()) r.metrics.RouterChannelQueueSend.Observe(time.Since(start).Seconds())
r.logger.Debug("received message", "peer", peerID, "message", msg) r.logger.Debug("received message", "peer", peerID, "message", msg)


Loading…
Cancel
Save