281 lines
6.8 KiB
Go
281 lines
6.8 KiB
Go
|
package gather
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"net/http"
|
||
|
"net/http/httptest"
|
||
|
"reflect"
|
||
|
"testing"
|
||
|
|
||
|
"github.com/google/go-cmp/cmp"
|
||
|
"github.com/influxdata/platform"
|
||
|
)
|
||
|
|
||
|
func TestPrometheusScraper(t *testing.T) {
|
||
|
cases := []struct {
|
||
|
name string
|
||
|
ms []Metrics
|
||
|
handler *mockHTTPHandler
|
||
|
hasErr bool
|
||
|
}{
|
||
|
{
|
||
|
name: "bad request",
|
||
|
hasErr: true,
|
||
|
},
|
||
|
{
|
||
|
name: "empty request",
|
||
|
handler: &mockHTTPHandler{
|
||
|
responseMap: map[string]string{
|
||
|
"/metrics": "",
|
||
|
},
|
||
|
},
|
||
|
hasErr: true,
|
||
|
},
|
||
|
{
|
||
|
name: "regular metrics",
|
||
|
handler: &mockHTTPHandler{
|
||
|
responseMap: map[string]string{
|
||
|
"/metrics": sampleResp,
|
||
|
},
|
||
|
},
|
||
|
ms: []Metrics{
|
||
|
{
|
||
|
Name: "go_gc_duration_seconds",
|
||
|
Type: MetricTypeSummary,
|
||
|
Fields: map[string]interface{}{
|
||
|
"count": float64(326),
|
||
|
"sum": 0.07497837,
|
||
|
"0": 3.6257e-05,
|
||
|
"0.25": 0.0001434,
|
||
|
"0.5": 0.000194491,
|
||
|
"0.75": 0.000270339,
|
||
|
"1": 0.000789365,
|
||
|
},
|
||
|
Tags: map[string]string{},
|
||
|
},
|
||
|
{
|
||
|
Name: "go_goroutines",
|
||
|
Type: MetricTypeGauge,
|
||
|
Tags: map[string]string{},
|
||
|
Fields: map[string]interface{}{
|
||
|
"gauge": float64(36),
|
||
|
},
|
||
|
},
|
||
|
{
|
||
|
Name: "go_info",
|
||
|
Type: MetricTypeGauge,
|
||
|
Tags: map[string]string{
|
||
|
"version": "go1.10.3",
|
||
|
},
|
||
|
Fields: map[string]interface{}{
|
||
|
"gauge": float64(1),
|
||
|
},
|
||
|
},
|
||
|
{
|
||
|
Name: "go_memstats_alloc_bytes",
|
||
|
Type: MetricTypeGauge,
|
||
|
Tags: map[string]string{},
|
||
|
Fields: map[string]interface{}{
|
||
|
"gauge": 2.0091312e+07,
|
||
|
},
|
||
|
},
|
||
|
{
|
||
|
Name: "go_memstats_alloc_bytes_total",
|
||
|
Type: MetricTypeCounter,
|
||
|
Fields: map[string]interface{}{
|
||
|
"counter": 4.183173328e+09,
|
||
|
},
|
||
|
Tags: map[string]string{},
|
||
|
},
|
||
|
{
|
||
|
Name: "go_memstats_buck_hash_sys_bytes",
|
||
|
Type: MetricTypeGauge,
|
||
|
Tags: map[string]string{},
|
||
|
Fields: map[string]interface{}{
|
||
|
"gauge": 1.533852e+06,
|
||
|
},
|
||
|
},
|
||
|
{
|
||
|
Name: "go_memstats_frees_total",
|
||
|
Type: MetricTypeCounter,
|
||
|
Tags: map[string]string{},
|
||
|
Fields: map[string]interface{}{
|
||
|
"counter": 1.8944339e+07,
|
||
|
},
|
||
|
},
|
||
|
{
|
||
|
Name: "go_memstats_gc_cpu_fraction",
|
||
|
Type: MetricTypeGauge,
|
||
|
Tags: map[string]string{},
|
||
|
Fields: map[string]interface{}{
|
||
|
"gauge": 1.972734963012756e-05,
|
||
|
},
|
||
|
},
|
||
|
},
|
||
|
hasErr: false,
|
||
|
},
|
||
|
}
|
||
|
for _, c := range cases {
|
||
|
scraper := new(prometheusScraper)
|
||
|
var url string
|
||
|
if c.handler != nil {
|
||
|
ts := httptest.NewServer(c.handler)
|
||
|
defer ts.Close()
|
||
|
url = ts.URL
|
||
|
}
|
||
|
results, err := scraper.Gather(context.Background(), platform.ScraperTarget{
|
||
|
URL: url + "/metrics",
|
||
|
OrgName: "org1",
|
||
|
BucketName: "bucket1",
|
||
|
})
|
||
|
if err != nil && !c.hasErr {
|
||
|
t.Fatalf("scraper parse err in testing %s: %v", c.name, err)
|
||
|
}
|
||
|
if len(c.ms) != len(results) {
|
||
|
t.Fatalf("scraper parse metrics incorrect length, want %d, got %d",
|
||
|
len(c.ms), len(results))
|
||
|
}
|
||
|
for _, m := range results {
|
||
|
for _, cm := range c.ms {
|
||
|
if m.Name == cm.Name {
|
||
|
if diff := cmp.Diff(m, cm, metricsCmpOption); diff != "" {
|
||
|
t.Fatalf("scraper parse metrics want %v, got %v", cm, m)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
const sampleResp = `
|
||
|
# HELP go_gc_duration_seconds A summary of the GC invocation durations.
|
||
|
# TYPE go_gc_duration_seconds summary
|
||
|
go_gc_duration_seconds{quantile="0"} 3.6257e-05
|
||
|
go_gc_duration_seconds{quantile="0.25"} 0.0001434
|
||
|
go_gc_duration_seconds{quantile="0.5"} 0.000194491
|
||
|
go_gc_duration_seconds{quantile="0.75"} 0.000270339
|
||
|
go_gc_duration_seconds{quantile="1"} 0.000789365
|
||
|
go_gc_duration_seconds_sum 0.07497837
|
||
|
go_gc_duration_seconds_count 326
|
||
|
# HELP go_goroutines Number of goroutines that currently exist.
|
||
|
# TYPE go_goroutines gauge
|
||
|
go_goroutines 36
|
||
|
# HELP go_info Information about the Go environment.
|
||
|
# TYPE go_info gauge
|
||
|
go_info{version="go1.10.3"} 1
|
||
|
# HELP go_memstats_alloc_bytes Number of bytes allocated and still in use.
|
||
|
# TYPE go_memstats_alloc_bytes gauge
|
||
|
go_memstats_alloc_bytes 2.0091312e+07
|
||
|
# HELP go_memstats_alloc_bytes_total Total number of bytes allocated, even if freed.
|
||
|
# TYPE go_memstats_alloc_bytes_total counter
|
||
|
go_memstats_alloc_bytes_total 4.183173328e+09
|
||
|
# HELP go_memstats_buck_hash_sys_bytes Number of bytes used by the profiling bucket hash table.
|
||
|
# TYPE go_memstats_buck_hash_sys_bytes gauge
|
||
|
go_memstats_buck_hash_sys_bytes 1.533852e+06
|
||
|
# HELP go_memstats_frees_total Total number of frees.
|
||
|
# TYPE go_memstats_frees_total counter
|
||
|
go_memstats_frees_total 1.8944339e+07
|
||
|
# HELP go_memstats_gc_cpu_fraction The fraction of this program's available CPU time used by the GC since the program started.
|
||
|
# TYPE go_memstats_gc_cpu_fraction gauge
|
||
|
go_memstats_gc_cpu_fraction 1.972734963012756e-05
|
||
|
`
|
||
|
|
||
|
// mockStorage implement storage interface
|
||
|
// and platform.ScraperTargetStoreService interface.
|
||
|
type mockStorage struct {
|
||
|
Metrics map[int64]Metrics
|
||
|
Targets []platform.ScraperTarget
|
||
|
}
|
||
|
|
||
|
func (s *mockStorage) Record(ms []Metrics) error {
|
||
|
for _, m := range ms {
|
||
|
s.Metrics[m.Timestamp] = m
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (s *mockStorage) ListTargets(ctx context.Context) (targets []platform.ScraperTarget, err error) {
|
||
|
if s.Targets == nil {
|
||
|
s.Targets = make([]platform.ScraperTarget, 0)
|
||
|
}
|
||
|
return s.Targets, nil
|
||
|
}
|
||
|
|
||
|
func (s *mockStorage) AddTarget(ctx context.Context, t *platform.ScraperTarget) error {
|
||
|
if s.Targets == nil {
|
||
|
s.Targets = make([]platform.ScraperTarget, 0)
|
||
|
}
|
||
|
s.Targets = append(s.Targets, *t)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (s *mockStorage) RemoveTarget(ctx context.Context, id platform.ID) error {
|
||
|
if s.Targets == nil {
|
||
|
return nil
|
||
|
}
|
||
|
for k, v := range s.Targets {
|
||
|
if v.ID.String() == id.String() {
|
||
|
s.Targets = append(s.Targets[:k], s.Targets[k+1:]...)
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (s *mockStorage) GetTargetByID(ctx context.Context, id platform.ID) (target *platform.ScraperTarget, err error) {
|
||
|
for k, v := range s.Targets {
|
||
|
if v.ID.String() == id.String() {
|
||
|
target = &s.Targets[k]
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return target, err
|
||
|
|
||
|
}
|
||
|
|
||
|
func (s *mockStorage) UpdateTarget(ctx context.Context, update *platform.ScraperTarget) (target *platform.ScraperTarget, err error) {
|
||
|
for k, v := range s.Targets {
|
||
|
if v.ID.String() == string(update.ID) {
|
||
|
s.Targets[k] = *update
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return update, err
|
||
|
}
|
||
|
|
||
|
type mockHTTPHandler struct {
|
||
|
unauthorized bool
|
||
|
noContent bool
|
||
|
responseMap map[string]string
|
||
|
}
|
||
|
|
||
|
func (h mockHTTPHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||
|
if h.unauthorized {
|
||
|
w.WriteHeader(http.StatusUnauthorized)
|
||
|
return
|
||
|
}
|
||
|
if h.noContent {
|
||
|
w.WriteHeader(http.StatusNoContent)
|
||
|
return
|
||
|
}
|
||
|
s, ok := h.responseMap[r.URL.Path]
|
||
|
if !ok {
|
||
|
w.WriteHeader(http.StatusNotFound)
|
||
|
return
|
||
|
}
|
||
|
w.Header().Set("Content-Type", "text/plain; version=0.0.4; charset=utf-8")
|
||
|
w.Write([]byte(s))
|
||
|
}
|
||
|
|
||
|
var metricsCmpOption = cmp.Options{
|
||
|
cmp.Comparer(func(x, y Metrics) bool {
|
||
|
return x.Name == y.Name &&
|
||
|
x.Type == y.Type &&
|
||
|
reflect.DeepEqual(x.Tags, y.Tags) &&
|
||
|
reflect.DeepEqual(x.Fields, y.Fields)
|
||
|
}),
|
||
|
}
|