Skip to content

Commit

Permalink
cdc/sink: kafka sink integrate broker level metrics (pingcap#4517)
Browse files Browse the repository at this point in the history
  • Loading branch information
3AceShowHand authored and zhaoxinyu committed Feb 16, 2022
1 parent e8897e3 commit d1ce271
Show file tree
Hide file tree
Showing 3 changed files with 1,406 additions and 478 deletions.
20 changes: 13 additions & 7 deletions cdc/sink/producer/kafka/kafka.go
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,7 @@ type kafkaSaramaProducer struct {
// Since we don't close these two clients (which have an input chan) from the
// sender routine, data race or send on closed chan could happen.
clientLock sync.RWMutex
admin kafka.ClusterAdminClient
client sarama.Client
asyncProducer sarama.AsyncProducer
syncProducer sarama.SyncProducer
Expand Down Expand Up @@ -262,6 +263,16 @@ func (k *kafkaSaramaProducer) Close() error {
zap.String("changefeed", k.id), zap.Any("role", k.role))
}

start = time.Now()
if err := k.admin.Close(); err != nil {
log.Warn("close kafka cluster admin with error", zap.Error(err),
zap.Duration("duration", time.Since(start)),
zap.String("changefeed", k.id), zap.Any("role", k.role))
} else {
log.Info("kafka cluster admin closed", zap.Duration("duration", time.Since(start)),
zap.String("changefeed", k.id), zap.Any("role", k.role))
}

k.metricsMonitor.Cleanup()
return nil
}
Expand Down Expand Up @@ -329,12 +340,6 @@ func NewKafkaSaramaProducer(ctx context.Context, topic string, config *Config,
if err != nil {
return nil, cerror.WrapError(cerror.ErrKafkaNewSaramaProducer, err)
}
defer func() {
if err := admin.Close(); err != nil {
log.Warn("close kafka cluster admin failed", zap.Error(err),
zap.String("changefeed", changefeedID), zap.Any("role", role))
}
}()

if err := validateAndCreateTopic(admin, topic, config, cfg, opts); err != nil {
return nil, cerror.WrapError(cerror.ErrKafkaNewSaramaProducer, err)
Expand All @@ -361,6 +366,7 @@ func NewKafkaSaramaProducer(ctx context.Context, topic string, config *Config,
return nil, err
}
k := &kafkaSaramaProducer{
admin: admin,
client: client,
asyncProducer: asyncProducer,
syncProducer: syncProducer,
Expand All @@ -380,7 +386,7 @@ func NewKafkaSaramaProducer(ctx context.Context, topic string, config *Config,
role: role,

metricsMonitor: NewSaramaMetricsMonitor(cfg.MetricRegistry,
util.CaptureAddrFromCtx(ctx), changefeedID),
util.CaptureAddrFromCtx(ctx), changefeedID, admin),
}
go func() {
if err := k.run(ctx); err != nil && errors.Cause(err) != context.Canceled {
Expand Down
209 changes: 201 additions & 8 deletions cdc/sink/producer/kafka/metrics.go
Original file line number Diff line number Diff line change
Expand Up @@ -14,12 +14,17 @@
package kafka

import (
"strconv"

"github.com/pingcap/log"
"github.com/pingcap/tiflow/pkg/kafka"
"github.com/prometheus/client_golang/prometheus"
"github.com/rcrowley/go-metrics"
"go.uber.org/zap"
)

var (
// batch-size
// Histogram update by the `batch-size`
batchSizeGauge = prometheus.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: "ticdc",
Expand All @@ -28,7 +33,7 @@ var (
Help: "the number of bytes sent per partition per request for all topics",
}, []string{"capture", "changefeed"})

// record-send-rate
// meter mark by total records count
recordSendRateGauge = prometheus.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: "ticdc",
Expand All @@ -38,6 +43,7 @@ var (
}, []string{"capture", "changefeed"})

// records-per-request
// histogram update by all records count.
recordPerRequestGauge = prometheus.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: "ticdc",
Expand All @@ -46,14 +52,88 @@ var (
Help: "the number of records sent per request for all topics",
}, []string{"capture", "changefeed"})

// compression-ratio
// histogram update by `compression-ratio`.
compressionRatioGauge = prometheus.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: "ticdc",
Subsystem: "sink",
Name: "kafka_producer_compression_ratio",
Help: "the compression ratio times 100 of record batches for all topics",
}, []string{"capture", "changefeed"})

// metrics for outgoing events
// meter mark for each request's size in bytes
outgoingByteRateGauge = prometheus.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: "ticdc",
Subsystem: "sink",
Name: "kafka_producer_outgoing_byte_rate",
Help: "Bytes/second written off all brokers",
}, []string{"capture", "changefeed", "broker"})

// meter mark by 1 for each request
requestRateGauge = prometheus.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: "ticdc",
Subsystem: "sink",
Name: "kafka_producer_request_rate",
Help: "Requests/second sent to all brokers",
}, []string{"capture", "changefeed", "broker"})

// meter mark for each request's size in bytes
requestSizeGauge = prometheus.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: "ticdc",
Subsystem: "sink",
Name: "kafka_producer_request_size",
Help: "the request size in bytes for all brokers",
}, []string{"capture", "changefeed", "broker"})

// histogram update for each received response, requestLatency := time.Since(response.requestTime)
requestLatencyInMsGauge = prometheus.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: "ticdc",
Subsystem: "sink",
Name: "kafka_producer_request_latency",
Help: "the request latency in ms for all brokers",
}, []string{"capture", "changefeed", "broker"})

// counter inc by 1 once a request send, dec by 1 for a response received.
requestsInFlightGauge = prometheus.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: "ticdc",
Subsystem: "sink",
Name: "kafka_producer_in_flight_requests",
Help: "the current number of in-flight requests awaiting a response for all brokers",
}, []string{"capture", "changefeed", "broker"})

// metrics for incoming events
// meter mark for each received response's size in bytes
incomingByteRateGauge = prometheus.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: "ticdc",
Subsystem: "sink",
Name: "kafka_producer_incoming_byte_rate",
Help: "Bytes/second read off all brokers",
}, []string{"capture", "changefeed", "broker"})

// meter mark by 1 once a response received.
responseRateGauge = prometheus.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: "ticdc",
Subsystem: "sink",
Name: "kafka_producer_response_rate",
Help: "Responses/second received from all brokers",
}, []string{"capture", "changefeed", "broker"})

// meter mark by each read response size
responseSizeGauge = prometheus.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: "ticdc",
Subsystem: "sink",
Name: "kafka_producer_response_size",
Help: "the response size in bytes for all brokers",
}, []string{"capture", "changefeed", "broker"})
)

// InitMetrics registers all metrics in this file
Expand All @@ -62,57 +142,170 @@ func InitMetrics(registry *prometheus.Registry) {
registry.MustRegister(recordSendRateGauge)
registry.MustRegister(recordPerRequestGauge)
registry.MustRegister(compressionRatioGauge)

registry.MustRegister(incomingByteRateGauge)
registry.MustRegister(outgoingByteRateGauge)
registry.MustRegister(requestSizeGauge)
registry.MustRegister(requestRateGauge)
registry.MustRegister(requestLatencyInMsGauge)
registry.MustRegister(requestsInFlightGauge)
registry.MustRegister(responseSizeGauge)
registry.MustRegister(responseRateGauge)
}

// sarama metrics names, see https://pkg.go.dev/github.com/Shopify/sarama#pkg-overview
const (
// metrics at producer level.
batchSizeMetricName = "batch-size"
recordSendRateMetricName = "record-send-rate"
recordPerRequestMetricName = "records-per-request"
compressionRatioMetricName = "compression-ratio"

// metrics at broker level.
incomingByteRateMetricNamePrefix = "incoming-byte-rate-for-broker-"
outgoingByteRateMetricNamePrefix = "outgoing-byte-rate-for-broker-"
requestRateMetricNamePrefix = "request-rate-for-broker-"
requestSizeMetricNamePrefix = "request-size-for-broker-"
requestLatencyInMsMetricNamePrefix = "request-latency-in-ms-for-broker-"
requestsInFlightMetricNamePrefix = "requests-in-flight-for-broker-"
responseRateMetricNamePrefix = "response-rate-for-broker-"
responseSizeMetricNamePrefix = "response-size-for-broker-"
)

type saramaMetricsMonitor struct {
captureAddr string
changefeedID string

registry metrics.Registry
admin kafka.ClusterAdminClient
}

// CollectMetrics collect all monitored metrics
func (sm *saramaMetricsMonitor) CollectMetrics() {
sm.collectProducerMetrics()
if err := sm.collectBrokerMetrics(); err != nil {
log.Warn("collect broker metrics failed", zap.Error(err))
}
}

func (sm *saramaMetricsMonitor) collectProducerMetrics() {
batchSizeMetric := sm.registry.Get(batchSizeMetricName)
if histogram, ok := batchSizeMetric.(metrics.Histogram); ok {
batchSizeGauge.WithLabelValues(sm.captureAddr, sm.changefeedID).Set(histogram.Mean())
batchSizeGauge.WithLabelValues(sm.captureAddr, sm.changefeedID).Set(histogram.Snapshot().Mean())
}

recordSendRateMetric := sm.registry.Get(recordSendRateMetricName)
if meter, ok := recordSendRateMetric.(metrics.Meter); ok {
recordSendRateGauge.WithLabelValues(sm.captureAddr, sm.changefeedID).Set(meter.Rate1())
recordSendRateGauge.WithLabelValues(sm.captureAddr, sm.changefeedID).Set(meter.Snapshot().Rate1())
}

recordPerRequestMetric := sm.registry.Get(recordPerRequestMetricName)
if histogram, ok := recordPerRequestMetric.(metrics.Histogram); ok {
recordPerRequestGauge.WithLabelValues(sm.captureAddr, sm.changefeedID).Set(histogram.Mean())
recordPerRequestGauge.WithLabelValues(sm.captureAddr, sm.changefeedID).Set(histogram.Snapshot().Mean())
}

compressionRatioMetric := sm.registry.Get(compressionRatioMetricName)
if histogram, ok := compressionRatioMetric.(metrics.Histogram); ok {
compressionRatioGauge.WithLabelValues(sm.captureAddr, sm.changefeedID).Set(histogram.Mean())
compressionRatioGauge.WithLabelValues(sm.captureAddr, sm.changefeedID).Set(histogram.Snapshot().Mean())
}
}

func NewSaramaMetricsMonitor(registry metrics.Registry, captureAddr, changefeedID string) *saramaMetricsMonitor {
func getBrokerMetricName(prefix, brokerID string) string {
return prefix + brokerID
}

func (sm *saramaMetricsMonitor) collectBrokerMetrics() error {
brokers, _, err := sm.admin.DescribeCluster()
if err != nil {
return err
}

for _, b := range brokers {
brokerID := strconv.Itoa(int(b.ID()))

incomingByteRateMetric := sm.registry.Get(getBrokerMetricName(incomingByteRateMetricNamePrefix, brokerID))
if meter, ok := incomingByteRateMetric.(metrics.Meter); ok {
incomingByteRateGauge.WithLabelValues(sm.captureAddr, sm.changefeedID, brokerID).Set(meter.Snapshot().Rate1())
}

outgoingByteRateMetric := sm.registry.Get(getBrokerMetricName(outgoingByteRateMetricNamePrefix, brokerID))
if meter, ok := outgoingByteRateMetric.(metrics.Meter); ok {
outgoingByteRateGauge.WithLabelValues(sm.captureAddr, sm.changefeedID, brokerID).Set(meter.Snapshot().Rate1())
}

requestRateMetric := sm.registry.Get(getBrokerMetricName(requestRateMetricNamePrefix, brokerID))
if meter, ok := requestRateMetric.(metrics.Meter); ok {
requestRateGauge.WithLabelValues(sm.captureAddr, sm.changefeedID, brokerID).Set(meter.Snapshot().Rate1())
}

requestSizeMetric := sm.registry.Get(getBrokerMetricName(requestSizeMetricNamePrefix, brokerID))
if histogram, ok := requestSizeMetric.(metrics.Histogram); ok {
requestSizeGauge.WithLabelValues(sm.captureAddr, sm.changefeedID, brokerID).Set(histogram.Snapshot().Mean())
}

requestLatencyMetric := sm.registry.Get(getBrokerMetricName(requestLatencyInMsMetricNamePrefix, brokerID))
if histogram, ok := requestLatencyMetric.(metrics.Histogram); ok {
requestLatencyInMsGauge.WithLabelValues(sm.captureAddr, sm.changefeedID, brokerID).Set(histogram.Snapshot().Mean())
}

requestsInFlightMetric := sm.registry.Get(getBrokerMetricName(requestsInFlightMetricNamePrefix, brokerID))
if counter, ok := requestsInFlightMetric.(metrics.Counter); ok {
requestsInFlightGauge.WithLabelValues(sm.captureAddr, sm.changefeedID, brokerID).Set(float64(counter.Snapshot().Count()))
}

responseRateMetric := sm.registry.Get(getBrokerMetricName(responseRateMetricNamePrefix, brokerID))
if meter, ok := responseRateMetric.(metrics.Meter); ok {
responseRateGauge.WithLabelValues(sm.captureAddr, sm.changefeedID, brokerID).Set(meter.Snapshot().Rate1())
}

responseSizeMetric := sm.registry.Get(getBrokerMetricName(responseSizeMetricNamePrefix, brokerID))
if histogram, ok := responseSizeMetric.(metrics.Histogram); ok {
responseSizeGauge.WithLabelValues(sm.captureAddr, sm.changefeedID, brokerID).Set(histogram.Snapshot().Mean())
}
}
return nil
}

func NewSaramaMetricsMonitor(registry metrics.Registry, captureAddr, changefeedID string, admin kafka.ClusterAdminClient) *saramaMetricsMonitor {
return &saramaMetricsMonitor{
captureAddr: captureAddr,
changefeedID: changefeedID,
registry: registry,
admin: admin,
}
}

func (sm *saramaMetricsMonitor) Cleanup() {
sm.cleanUpProducerMetrics()
if err := sm.cleanUpBrokerMetrics(); err != nil {
log.Warn("clean up broker metrics failed", zap.Error(err))
}
}

func (sm *saramaMetricsMonitor) cleanUpProducerMetrics() {
batchSizeGauge.DeleteLabelValues(sm.captureAddr, sm.changefeedID)
recordSendRateGauge.DeleteLabelValues(sm.captureAddr, sm.changefeedID)
recordPerRequestGauge.DeleteLabelValues(sm.captureAddr, sm.changefeedID)
compressionRatioGauge.DeleteLabelValues(sm.captureAddr, sm.changefeedID)
}

func (sm *saramaMetricsMonitor) cleanUpBrokerMetrics() error {
brokers, _, err := sm.admin.DescribeCluster()
if err != nil {
return err
}

for _, b := range brokers {
brokerID := strconv.Itoa(int(b.ID()))

incomingByteRateGauge.DeleteLabelValues(sm.captureAddr, sm.changefeedID, brokerID)
outgoingByteRateGauge.DeleteLabelValues(sm.captureAddr, sm.changefeedID, brokerID)
requestRateGauge.DeleteLabelValues(sm.captureAddr, sm.changefeedID, brokerID)
requestSizeGauge.DeleteLabelValues(sm.captureAddr, sm.changefeedID, brokerID)
requestLatencyInMsGauge.DeleteLabelValues(sm.captureAddr, sm.changefeedID, brokerID)
requestsInFlightGauge.DeleteLabelValues(sm.captureAddr, sm.changefeedID, brokerID)
responseRateGauge.DeleteLabelValues(sm.captureAddr, sm.changefeedID, brokerID)
responseSizeGauge.DeleteLabelValues(sm.captureAddr, sm.changefeedID, brokerID)
}
return nil
}
Loading

0 comments on commit d1ce271

Please sign in to comment.