Skip to content

Commit

Permalink
internalstorage: add gorm database metrics
Browse files Browse the repository at this point in the history
Signed-off-by: Iceber Gu <[email protected]>
  • Loading branch information
Iceber committed Dec 19, 2024
1 parent 44280d6 commit 17c43f4
Show file tree
Hide file tree
Showing 2 changed files with 184 additions and 0 deletions.
178 changes: 178 additions & 0 deletions pkg/storage/internalstorage/metrics.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,178 @@
package internalstorage

/*
Copy from https://github.com/go-gorm/prometheus
Some streamlining and optimization as features like Push and HTTPServer are not needed.
*/

import (
"context"
"database/sql"
"reflect"
"sync"
"time"

"github.com/prometheus/client_golang/prometheus"
"gorm.io/gorm"
"k8s.io/component-base/metrics/legacyregistry"
)

var (
_ gorm.Plugin = &Prometheus{}
)

const (
defaultRefreshInterval = 15 // the prometheus default pull metrics every 15 seconds
)

type Prometheus struct {
*gorm.DB
*DBStats

refreshInterval uint32
refreshOnce sync.Once
Labels map[string]string
}

type MetricsConfig struct {
DBName string // use DBName as metrics label
RefreshInterval uint32 // refresh metrics interval.
Labels map[string]string // metrics labels
}

func NewGormMetrics(config MetricsConfig) *Prometheus {
if config.RefreshInterval == 0 {
config.RefreshInterval = defaultRefreshInterval
}

labels := make(map[string]string)
if config.Labels != nil {
labels = config.Labels
}
if config.DBName != "" {
labels["db_name"] = config.DBName
}

return &Prometheus{refreshInterval: config.RefreshInterval, Labels: labels}
}

func (p *Prometheus) Name() string {
return "gorm:prometheus"
}

func (p *Prometheus) Initialize(db *gorm.DB) error { // can be called repeatedly
p.DB = db
p.DBStats = newStats(p.Labels)

p.refreshOnce.Do(func() {
go func() {
for range time.Tick(time.Duration(p.refreshInterval) * time.Second) {
p.refresh()
}
}()
})
return nil
}

func (p *Prometheus) refresh() {
db, err := p.DB.DB()
if err != nil {
p.DB.Logger.Error(context.Background(), "gorm:prometheus failed to collect db status, got error: %v", err)
return
}
p.DBStats.set(db.Stats())
}

type DBStats struct {
MaxOpenConnections prometheus.Gauge // Maximum number of open connections to the database.

// Pool status
OpenConnections prometheus.Gauge // The number of established connections both in use and idle.
InUse prometheus.Gauge // The number of connections currently in use.
Idle prometheus.Gauge // The number of idle connections.

// Counters
WaitCount prometheus.Gauge // The total number of connections waited for.
WaitDuration prometheus.Gauge // The total time blocked waiting for a new connection.
MaxIdleClosed prometheus.Gauge // The total number of connections closed due to SetMaxIdleConns.
MaxLifetimeClosed prometheus.Gauge // The total number of connections closed due to SetConnMaxLifetime.
MaxIdleTimeClosed prometheus.Gauge // The total number of connections closed due to SetConnMaxIdleTime.
}

func newStats(labels map[string]string) *DBStats {
stats := &DBStats{
MaxOpenConnections: prometheus.NewGauge(prometheus.GaugeOpts{
Name: "gorm_dbstats_max_open_connections",
Help: "Maximum number of open connections to the database.",
ConstLabels: labels,
}),
OpenConnections: prometheus.NewGauge(prometheus.GaugeOpts{
Name: "gorm_dbstats_open_connections",
Help: "The number of established connections both in use and idle.",
ConstLabels: labels,
}),
InUse: prometheus.NewGauge(prometheus.GaugeOpts{
Name: "gorm_dbstats_in_use",
Help: "The number of connections currently in use.",
ConstLabels: labels,
}),
Idle: prometheus.NewGauge(prometheus.GaugeOpts{
Name: "gorm_dbstats_idle",
Help: "The number of idle connections.",
ConstLabels: labels,
}),
WaitCount: prometheus.NewGauge(prometheus.GaugeOpts{
Name: "gorm_dbstats_wait_count",
Help: "The total number of connections waited for.",
ConstLabels: labels,
}),
WaitDuration: prometheus.NewGauge(prometheus.GaugeOpts{
Name: "gorm_dbstats_wait_duration",
Help: "The total time blocked waiting for a new connection.",
ConstLabels: labels,
}),
MaxIdleClosed: prometheus.NewGauge(prometheus.GaugeOpts{
Name: "gorm_dbstats_max_idle_closed",
Help: "The total number of connections closed due to SetMaxIdleConns.",
ConstLabels: labels,
}),
MaxLifetimeClosed: prometheus.NewGauge(prometheus.GaugeOpts{
Name: "gorm_dbstats_max_lifetime_closed",
Help: "The total number of connections closed due to SetConnMaxLifetime.",
ConstLabels: labels,
}),
MaxIdleTimeClosed: prometheus.NewGauge(prometheus.GaugeOpts{
Name: "gorm_dbstats_max_idletime_closed",
Help: "The total number of connections closed due to SetConnMaxIdleTime.",
ConstLabels: labels,
}),
}

for _, collector := range stats.collectors() {
legacyregistry.RawMustRegister(collector)
}

return stats
}

func (stats *DBStats) set(dbStats sql.DBStats) {
stats.MaxOpenConnections.Set(float64(dbStats.MaxOpenConnections))
stats.OpenConnections.Set(float64(dbStats.OpenConnections))
stats.InUse.Set(float64(dbStats.InUse))
stats.Idle.Set(float64(dbStats.Idle))
stats.WaitCount.Set(float64(dbStats.WaitCount))
stats.WaitDuration.Set(float64(dbStats.WaitDuration))
stats.MaxIdleClosed.Set(float64(dbStats.MaxIdleClosed))
stats.MaxLifetimeClosed.Set(float64(dbStats.MaxLifetimeClosed))
stats.MaxIdleTimeClosed.Set(float64(dbStats.MaxIdleTimeClosed))
}

// get collector in stats
func (stats *DBStats) collectors() (collector []prometheus.Collector) {
dbStatsValue := reflect.ValueOf(*stats)
for i := 0; i < dbStatsValue.NumField(); i++ {
collector = append(collector, dbStatsValue.Field(i).Interface().(prometheus.Gauge))
}
return
}
6 changes: 6 additions & 0 deletions pkg/storage/internalstorage/register.go
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,12 @@ func NewStorageFactory(configPath string) (storage.StorageFactory, error) {
if err != nil {
return nil, err
}

sqlProme := NewGormMetrics(MetricsConfig{DBName: cfg.Database})
if err := db.Use(sqlProme); err != nil {
return nil, err
}

sqlDB, err := db.DB()
if err != nil {
return nil, err
Expand Down

0 comments on commit 17c43f4

Please sign in to comment.