feat: add metrics (#3624)
parent
199e86050e
commit
c05e03bb5a
@ -0,0 +1,130 @@
|
|||||||
|
package redis
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
red "github.com/go-redis/redis/v8"
|
||||||
|
"github.com/prometheus/client_golang/prometheus"
|
||||||
|
"github.com/prometheus/client_golang/prometheus/testutil"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/zeromicro/go-zero/core/conf"
|
||||||
|
"github.com/zeromicro/go-zero/internal/devserver"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestRedisMetric(t *testing.T) {
|
||||||
|
cfg := devserver.Config{}
|
||||||
|
_ = conf.FillDefault(&cfg)
|
||||||
|
server := devserver.NewServer(cfg)
|
||||||
|
server.StartAsync()
|
||||||
|
time.Sleep(time.Second)
|
||||||
|
|
||||||
|
metricReqDur.Observe(8, "test-cmd")
|
||||||
|
metricReqErr.Inc("test-cmd", "internal-error")
|
||||||
|
metricSlowCount.Inc("test-cmd")
|
||||||
|
|
||||||
|
url := "http://127.0.0.1:6060/metrics"
|
||||||
|
resp, err := http.Get(url)
|
||||||
|
assert.Nil(t, err)
|
||||||
|
defer resp.Body.Close()
|
||||||
|
s, err := io.ReadAll(resp.Body)
|
||||||
|
assert.Nil(t, err)
|
||||||
|
content := string(s)
|
||||||
|
assert.Contains(t, content, "redis_client_requests_duration_ms_sum{command=\"test-cmd\"} 8\n")
|
||||||
|
assert.Contains(t, content, "redis_client_requests_duration_ms_count{command=\"test-cmd\"} 1\n")
|
||||||
|
assert.Contains(t, content, "redis_client_requests_error_total{command=\"test-cmd\",error=\"internal-error\"} 1\n")
|
||||||
|
assert.Contains(t, content, "redis_client_requests_slow_total{command=\"test-cmd\"} 1\n")
|
||||||
|
}
|
||||||
|
|
||||||
|
func Test_newCollector(t *testing.T) {
|
||||||
|
prometheus.Unregister(connCollector)
|
||||||
|
c := newCollector()
|
||||||
|
c.registerClient(&statGetter{
|
||||||
|
clientType: "node",
|
||||||
|
key: "test1",
|
||||||
|
poolSize: 10,
|
||||||
|
poolStats: func() *red.PoolStats {
|
||||||
|
return &red.PoolStats{
|
||||||
|
Hits: 10000,
|
||||||
|
Misses: 10,
|
||||||
|
Timeouts: 5,
|
||||||
|
TotalConns: 100,
|
||||||
|
IdleConns: 20,
|
||||||
|
StaleConns: 1,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
c.registerClient(&statGetter{
|
||||||
|
clientType: "node",
|
||||||
|
key: "test2",
|
||||||
|
poolSize: 11,
|
||||||
|
poolStats: func() *red.PoolStats {
|
||||||
|
return &red.PoolStats{
|
||||||
|
Hits: 10001,
|
||||||
|
Misses: 11,
|
||||||
|
Timeouts: 6,
|
||||||
|
TotalConns: 101,
|
||||||
|
IdleConns: 21,
|
||||||
|
StaleConns: 2,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
c.registerClient(&statGetter{
|
||||||
|
clientType: "cluster",
|
||||||
|
key: "test3",
|
||||||
|
poolSize: 5,
|
||||||
|
poolStats: func() *red.PoolStats {
|
||||||
|
return &red.PoolStats{
|
||||||
|
Hits: 20000,
|
||||||
|
Misses: 20,
|
||||||
|
Timeouts: 10,
|
||||||
|
TotalConns: 200,
|
||||||
|
IdleConns: 40,
|
||||||
|
StaleConns: 2,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
val := `
|
||||||
|
# HELP redis_client_pool_conn_idle_current Current number of idle connections in the pool
|
||||||
|
# TYPE redis_client_pool_conn_idle_current gauge
|
||||||
|
redis_client_pool_conn_idle_current{client_type="cluster",key="test3"} 40
|
||||||
|
redis_client_pool_conn_idle_current{client_type="node",key="test1"} 20
|
||||||
|
redis_client_pool_conn_idle_current{client_type="node",key="test2"} 21
|
||||||
|
# HELP redis_client_pool_conn_max Max number of connections in the pool
|
||||||
|
# TYPE redis_client_pool_conn_max counter
|
||||||
|
redis_client_pool_conn_max{client_type="cluster",key="test3"} 5
|
||||||
|
redis_client_pool_conn_max{client_type="node",key="test1"} 10
|
||||||
|
redis_client_pool_conn_max{client_type="node",key="test2"} 11
|
||||||
|
# HELP redis_client_pool_conn_stale_total Number of times a connection was removed from the pool because it was stale
|
||||||
|
# TYPE redis_client_pool_conn_stale_total counter
|
||||||
|
redis_client_pool_conn_stale_total{client_type="cluster",key="test3"} 2
|
||||||
|
redis_client_pool_conn_stale_total{client_type="node",key="test1"} 1
|
||||||
|
redis_client_pool_conn_stale_total{client_type="node",key="test2"} 2
|
||||||
|
# HELP redis_client_pool_conn_total_current Current number of connections in the pool
|
||||||
|
# TYPE redis_client_pool_conn_total_current gauge
|
||||||
|
redis_client_pool_conn_total_current{client_type="cluster",key="test3"} 200
|
||||||
|
redis_client_pool_conn_total_current{client_type="node",key="test1"} 100
|
||||||
|
redis_client_pool_conn_total_current{client_type="node",key="test2"} 101
|
||||||
|
# HELP redis_client_pool_hit_total Number of times a connection was found in the pool
|
||||||
|
# TYPE redis_client_pool_hit_total counter
|
||||||
|
redis_client_pool_hit_total{client_type="cluster",key="test3"} 20000
|
||||||
|
redis_client_pool_hit_total{client_type="node",key="test1"} 10000
|
||||||
|
redis_client_pool_hit_total{client_type="node",key="test2"} 10001
|
||||||
|
# HELP redis_client_pool_miss_total Number of times a connection was not found in the pool
|
||||||
|
# TYPE redis_client_pool_miss_total counter
|
||||||
|
redis_client_pool_miss_total{client_type="cluster",key="test3"} 20
|
||||||
|
redis_client_pool_miss_total{client_type="node",key="test1"} 10
|
||||||
|
redis_client_pool_miss_total{client_type="node",key="test2"} 11
|
||||||
|
# HELP redis_client_pool_timeout_total Number of times a timeout occurred when looking for a connection in the pool
|
||||||
|
# TYPE redis_client_pool_timeout_total counter
|
||||||
|
redis_client_pool_timeout_total{client_type="cluster",key="test3"} 10
|
||||||
|
redis_client_pool_timeout_total{client_type="node",key="test1"} 5
|
||||||
|
redis_client_pool_timeout_total{client_type="node",key="test2"} 6
|
||||||
|
`
|
||||||
|
|
||||||
|
err := testutil.CollectAndCompare(c, strings.NewReader(val))
|
||||||
|
assert.NoError(t, err)
|
||||||
|
}
|
@ -0,0 +1,147 @@
|
|||||||
|
package sqlx
|
||||||
|
|
||||||
|
import (
|
||||||
|
"database/sql"
|
||||||
|
"io"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/prometheus/client_golang/prometheus"
|
||||||
|
"github.com/prometheus/client_golang/prometheus/testutil"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/zeromicro/go-zero/core/conf"
|
||||||
|
"github.com/zeromicro/go-zero/internal/devserver"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestSqlxMetric(t *testing.T) {
|
||||||
|
cfg := devserver.Config{}
|
||||||
|
_ = conf.FillDefault(&cfg)
|
||||||
|
cfg.Port = 6480
|
||||||
|
server := devserver.NewServer(cfg)
|
||||||
|
server.StartAsync()
|
||||||
|
time.Sleep(time.Second)
|
||||||
|
|
||||||
|
metricReqDur.Observe(8, "test-cmd")
|
||||||
|
metricReqErr.Inc("test-cmd", "internal-error")
|
||||||
|
metricSlowCount.Inc("test-cmd")
|
||||||
|
|
||||||
|
url := "http://127.0.0.1:6480/metrics"
|
||||||
|
resp, err := http.Get(url)
|
||||||
|
assert.Nil(t, err)
|
||||||
|
defer resp.Body.Close()
|
||||||
|
s, err := io.ReadAll(resp.Body)
|
||||||
|
assert.Nil(t, err)
|
||||||
|
content := string(s)
|
||||||
|
assert.Contains(t, content, "mysql_client_requests_duration_ms_sum{command=\"test-cmd\"} 8\n")
|
||||||
|
assert.Contains(t, content, "mysql_client_requests_duration_ms_count{command=\"test-cmd\"} 1\n")
|
||||||
|
assert.Contains(t, content, "mysql_client_requests_error_total{command=\"test-cmd\",error=\"internal-error\"} 1\n")
|
||||||
|
assert.Contains(t, content, "mysql_client_requests_slow_total{command=\"test-cmd\"} 1\n")
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMetricCollector(t *testing.T) {
|
||||||
|
prometheus.Unregister(connCollector)
|
||||||
|
c := newCollector()
|
||||||
|
c.registerClient(&statGetter{
|
||||||
|
dbName: "db-1",
|
||||||
|
hash: "hash-1",
|
||||||
|
poolStats: func() sql.DBStats {
|
||||||
|
return sql.DBStats{
|
||||||
|
MaxOpenConnections: 1,
|
||||||
|
OpenConnections: 2,
|
||||||
|
InUse: 3,
|
||||||
|
Idle: 4,
|
||||||
|
WaitCount: 5,
|
||||||
|
WaitDuration: 6 * time.Second,
|
||||||
|
MaxIdleClosed: 7,
|
||||||
|
MaxIdleTimeClosed: 8,
|
||||||
|
MaxLifetimeClosed: 9,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
c.registerClient(&statGetter{
|
||||||
|
dbName: "db-1",
|
||||||
|
hash: "hash-2",
|
||||||
|
poolStats: func() sql.DBStats {
|
||||||
|
return sql.DBStats{
|
||||||
|
MaxOpenConnections: 10,
|
||||||
|
OpenConnections: 20,
|
||||||
|
InUse: 30,
|
||||||
|
Idle: 40,
|
||||||
|
WaitCount: 50,
|
||||||
|
WaitDuration: 60 * time.Second,
|
||||||
|
MaxIdleClosed: 70,
|
||||||
|
MaxIdleTimeClosed: 80,
|
||||||
|
MaxLifetimeClosed: 90,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
c.registerClient(&statGetter{
|
||||||
|
dbName: "db-2",
|
||||||
|
hash: "hash-2",
|
||||||
|
poolStats: func() sql.DBStats {
|
||||||
|
return sql.DBStats{
|
||||||
|
MaxOpenConnections: 100,
|
||||||
|
OpenConnections: 200,
|
||||||
|
InUse: 300,
|
||||||
|
Idle: 400,
|
||||||
|
WaitCount: 500,
|
||||||
|
WaitDuration: 600 * time.Second,
|
||||||
|
MaxIdleClosed: 700,
|
||||||
|
MaxIdleTimeClosed: 800,
|
||||||
|
MaxLifetimeClosed: 900,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
val := `
|
||||||
|
# HELP mysql_client_idle_connections The number of idle connections.
|
||||||
|
# TYPE mysql_client_idle_connections gauge
|
||||||
|
mysql_client_idle_connections{db_name="db-1",hash="hash-1"} 4
|
||||||
|
mysql_client_idle_connections{db_name="db-1",hash="hash-2"} 40
|
||||||
|
mysql_client_idle_connections{db_name="db-2",hash="hash-2"} 400
|
||||||
|
# HELP mysql_client_in_use_connections The number of connections currently in use.
|
||||||
|
# TYPE mysql_client_in_use_connections gauge
|
||||||
|
mysql_client_in_use_connections{db_name="db-1",hash="hash-1"} 3
|
||||||
|
mysql_client_in_use_connections{db_name="db-1",hash="hash-2"} 30
|
||||||
|
mysql_client_in_use_connections{db_name="db-2",hash="hash-2"} 300
|
||||||
|
# HELP mysql_client_max_idle_closed_total The total number of connections closed due to SetMaxIdleConns.
|
||||||
|
# TYPE mysql_client_max_idle_closed_total counter
|
||||||
|
mysql_client_max_idle_closed_total{db_name="db-1",hash="hash-1"} 7
|
||||||
|
mysql_client_max_idle_closed_total{db_name="db-1",hash="hash-2"} 70
|
||||||
|
mysql_client_max_idle_closed_total{db_name="db-2",hash="hash-2"} 700
|
||||||
|
# HELP mysql_client_max_idle_time_closed_total The total number of connections closed due to SetConnMaxIdleTime.
|
||||||
|
# TYPE mysql_client_max_idle_time_closed_total counter
|
||||||
|
mysql_client_max_idle_time_closed_total{db_name="db-1",hash="hash-1"} 8
|
||||||
|
mysql_client_max_idle_time_closed_total{db_name="db-1",hash="hash-2"} 80
|
||||||
|
mysql_client_max_idle_time_closed_total{db_name="db-2",hash="hash-2"} 800
|
||||||
|
# HELP mysql_client_max_lifetime_closed_total The total number of connections closed due to SetConnMaxLifetime.
|
||||||
|
# TYPE mysql_client_max_lifetime_closed_total counter
|
||||||
|
mysql_client_max_lifetime_closed_total{db_name="db-1",hash="hash-1"} 9
|
||||||
|
mysql_client_max_lifetime_closed_total{db_name="db-1",hash="hash-2"} 90
|
||||||
|
mysql_client_max_lifetime_closed_total{db_name="db-2",hash="hash-2"} 900
|
||||||
|
# HELP mysql_client_max_open_connections Maximum number of open connections to the database.
|
||||||
|
# TYPE mysql_client_max_open_connections gauge
|
||||||
|
mysql_client_max_open_connections{db_name="db-1",hash="hash-1"} 1
|
||||||
|
mysql_client_max_open_connections{db_name="db-1",hash="hash-2"} 10
|
||||||
|
mysql_client_max_open_connections{db_name="db-2",hash="hash-2"} 100
|
||||||
|
# HELP mysql_client_open_connections The number of established connections both in use and idle.
|
||||||
|
# TYPE mysql_client_open_connections gauge
|
||||||
|
mysql_client_open_connections{db_name="db-1",hash="hash-1"} 2
|
||||||
|
mysql_client_open_connections{db_name="db-1",hash="hash-2"} 20
|
||||||
|
mysql_client_open_connections{db_name="db-2",hash="hash-2"} 200
|
||||||
|
# HELP mysql_client_wait_count_total The total number of connections waited for.
|
||||||
|
# TYPE mysql_client_wait_count_total counter
|
||||||
|
mysql_client_wait_count_total{db_name="db-1",hash="hash-1"} 5
|
||||||
|
mysql_client_wait_count_total{db_name="db-1",hash="hash-2"} 50
|
||||||
|
mysql_client_wait_count_total{db_name="db-2",hash="hash-2"} 500
|
||||||
|
# HELP mysql_client_wait_duration_seconds_total The total time blocked waiting for a new connection.
|
||||||
|
# TYPE mysql_client_wait_duration_seconds_total counter
|
||||||
|
mysql_client_wait_duration_seconds_total{db_name="db-1",hash="hash-1"} 6
|
||||||
|
mysql_client_wait_duration_seconds_total{db_name="db-1",hash="hash-2"} 60
|
||||||
|
mysql_client_wait_duration_seconds_total{db_name="db-2",hash="hash-2"} 600
|
||||||
|
`
|
||||||
|
|
||||||
|
err := testutil.CollectAndCompare(c, strings.NewReader(val))
|
||||||
|
assert.NoError(t, err)
|
||||||
|
}
|
@ -0,0 +1,71 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
"strconv"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/zeromicro/go-zero/core/metric"
|
||||||
|
"github.com/zeromicro/go-zero/core/timex"
|
||||||
|
)
|
||||||
|
|
||||||
|
const clientNamespace = "httpc_client"
|
||||||
|
|
||||||
|
var (
|
||||||
|
MetricClientReqDur = metric.NewHistogramVec(&metric.HistogramVecOpts{
|
||||||
|
Namespace: clientNamespace,
|
||||||
|
Subsystem: "requests",
|
||||||
|
Name: "duration_ms",
|
||||||
|
Help: "http client requests duration(ms).",
|
||||||
|
Labels: []string{"name", "method", "url"},
|
||||||
|
Buckets: []float64{0.25, 0.5, 1, 2, 5, 10, 25, 50, 100, 250, 500, 1000, 2000, 5000, 10000, 15000},
|
||||||
|
})
|
||||||
|
|
||||||
|
MetricClientReqCodeTotal = metric.NewCounterVec(&metric.CounterVecOpts{
|
||||||
|
Namespace: clientNamespace,
|
||||||
|
Subsystem: "requests",
|
||||||
|
Name: "code_total",
|
||||||
|
Help: "http client requests code count.",
|
||||||
|
Labels: []string{"name", "method", "url", "code"},
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
|
type (
|
||||||
|
MetricsURLRewriter func(u url.URL) string
|
||||||
|
)
|
||||||
|
|
||||||
|
func MetricsInterceptor(name string, pr MetricsURLRewriter) Interceptor {
|
||||||
|
return func(r *http.Request) (*http.Request, ResponseHandler) {
|
||||||
|
startTime := timex.Now()
|
||||||
|
return r, func(resp *http.Response, err error) {
|
||||||
|
u := cleanURL(*r.URL)
|
||||||
|
method := r.Method
|
||||||
|
var (
|
||||||
|
code int
|
||||||
|
path string
|
||||||
|
)
|
||||||
|
// error or resp is nil, set code=500
|
||||||
|
if err != nil || resp == nil {
|
||||||
|
code = http.StatusInternalServerError
|
||||||
|
} else {
|
||||||
|
code = resp.StatusCode
|
||||||
|
}
|
||||||
|
if pr != nil {
|
||||||
|
path = pr(u)
|
||||||
|
} else {
|
||||||
|
path = u.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
MetricClientReqDur.ObserveFloat(float64(timex.Since(startTime))/float64(time.Millisecond), name, method, path)
|
||||||
|
MetricClientReqCodeTotal.Inc(name, method, path, strconv.Itoa(code))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func cleanURL(r url.URL) url.URL {
|
||||||
|
r.RawQuery = ""
|
||||||
|
r.RawFragment = ""
|
||||||
|
r.User = nil
|
||||||
|
return r
|
||||||
|
}
|
@ -0,0 +1,35 @@
|
|||||||
|
package internal
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/golang/mock/gomock"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"github.com/zeromicro/go-zero/core/logx"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestMetricsInterceptor(t *testing.T) {
|
||||||
|
c := gomock.NewController(t)
|
||||||
|
defer c.Finish()
|
||||||
|
|
||||||
|
logx.Disable()
|
||||||
|
|
||||||
|
svr := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
time.Sleep(100 * time.Millisecond)
|
||||||
|
w.WriteHeader(http.StatusInternalServerError)
|
||||||
|
}))
|
||||||
|
defer svr.Close()
|
||||||
|
|
||||||
|
req, err := http.NewRequest(http.MethodGet, svr.URL, nil)
|
||||||
|
assert.NotNil(t, req)
|
||||||
|
assert.Nil(t, err)
|
||||||
|
interceptor := MetricsInterceptor("test", nil)
|
||||||
|
req, handler := interceptor(req)
|
||||||
|
resp, err := http.DefaultClient.Do(req)
|
||||||
|
assert.NotNil(t, resp)
|
||||||
|
assert.Nil(t, err)
|
||||||
|
handler(resp, err)
|
||||||
|
}
|
Loading…
Reference in New Issue