mirror of
https://github.com/cuigh/swirl
synced 2024-12-31 16:23:13 +00:00
204 lines
4.4 KiB
Go
204 lines
4.4 KiB
Go
package biz
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"time"
|
|
|
|
"github.com/cuigh/auxo/ext/times"
|
|
"github.com/cuigh/auxo/util/cast"
|
|
"github.com/cuigh/auxo/util/lazy"
|
|
"github.com/cuigh/swirl/misc"
|
|
client "github.com/prometheus/client_golang/api"
|
|
papi "github.com/prometheus/client_golang/api/prometheus/v1"
|
|
"github.com/prometheus/common/model"
|
|
)
|
|
|
|
type MetricBiz interface {
|
|
Enabled() bool
|
|
GetMatrix(ctx context.Context, query, legend string, start, end time.Time) (data *MatrixData, err error)
|
|
GetScalar(ctx context.Context, query string, t time.Time) (data float64, err error)
|
|
GetVector(ctx context.Context, query, label string, t time.Time) (data *VectorData, err error)
|
|
}
|
|
|
|
func NewMetric(setting *misc.Setting) MetricBiz {
|
|
b := &metricBiz{prometheus: setting.Metric.Prometheus}
|
|
b.api.New = b.createAPI
|
|
return b
|
|
}
|
|
|
|
type metricBiz struct {
|
|
prometheus string
|
|
api lazy.Value
|
|
}
|
|
|
|
func (b *metricBiz) createAPI() (api interface{}, err error) {
|
|
if b.prometheus == "" {
|
|
return nil, errors.New("prometheus address is not configured")
|
|
}
|
|
|
|
var c client.Client
|
|
if c, err = client.NewClient(client.Config{Address: b.prometheus}); err == nil {
|
|
api = papi.NewAPI(c)
|
|
}
|
|
return
|
|
}
|
|
|
|
func (b *metricBiz) Enabled() bool {
|
|
return b.prometheus != ""
|
|
}
|
|
|
|
func (b *metricBiz) GetMatrix(ctx context.Context, query, legend string, start, end time.Time) (data *MatrixData, err error) {
|
|
if !b.Enabled() {
|
|
return
|
|
}
|
|
|
|
api, err := b.getAPI()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
period := end.Sub(start)
|
|
value, _, err := api.QueryRange(ctx, query, papi.Range{
|
|
Start: start,
|
|
End: end,
|
|
Step: b.calcStep(period),
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
data = &MatrixData{}
|
|
matrix := value.(model.Matrix)
|
|
for _, stream := range matrix {
|
|
data.Legend = append(data.Legend, b.formatLabel(legend, stream.Metric))
|
|
line := MatrixLine{Name: b.formatLabel(legend, stream.Metric)}
|
|
for _, v := range stream.Values {
|
|
p := MatrixPoint{
|
|
X: int64(v.Timestamp),
|
|
Y: float64(v.Value),
|
|
}
|
|
line.Data = append(line.Data, p)
|
|
}
|
|
data.Series = append(data.Series, line)
|
|
}
|
|
return
|
|
}
|
|
|
|
func (b *metricBiz) GetScalar(ctx context.Context, query string, t time.Time) (v float64, err error) {
|
|
if !b.Enabled() {
|
|
return
|
|
}
|
|
|
|
api, err := b.getAPI()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
value, _, err := api.Query(ctx, query, t)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
//scalar := value.(*model.Scalar)
|
|
vector := value.(model.Vector)
|
|
if len(vector) > 0 {
|
|
sample := vector[0]
|
|
return float64(sample.Value), nil
|
|
}
|
|
return 0, nil
|
|
}
|
|
|
|
func (b *metricBiz) GetVector(ctx context.Context, query, label string, t time.Time) (data *VectorData, err error) {
|
|
if !b.Enabled() {
|
|
return
|
|
}
|
|
|
|
var api papi.API
|
|
api, err = b.getAPI()
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
var value model.Value
|
|
value, _, err = api.Query(ctx, query, t)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
data = &VectorData{}
|
|
vector := value.(model.Vector)
|
|
for _, sample := range vector {
|
|
cv := VectorValue{
|
|
Name: b.formatLabel(label, sample.Metric),
|
|
Value: float64(sample.Value),
|
|
}
|
|
data.Data = append(data.Data, cv)
|
|
data.Legend = append(data.Legend, cv.Name)
|
|
}
|
|
return
|
|
}
|
|
|
|
func (b *metricBiz) calcStep(period time.Duration) (step time.Duration) {
|
|
if period >= times.Day {
|
|
step = 8 * time.Minute
|
|
} else if period >= 12*time.Hour {
|
|
step = 4 * time.Minute
|
|
} else if period >= 6*time.Hour {
|
|
step = 2 * time.Minute
|
|
} else if period >= 3*time.Hour {
|
|
step = time.Minute
|
|
} else {
|
|
step = 30 * time.Second
|
|
}
|
|
return
|
|
}
|
|
|
|
func (b *metricBiz) getAPI() (api papi.API, err error) {
|
|
v, err := b.api.Get()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return v.(papi.API), nil
|
|
}
|
|
|
|
func (b *metricBiz) formatLabel(label string, metric model.Metric) string {
|
|
return os.Expand(label, func(key string) string {
|
|
if s := string(metric[model.LabelName(key)]); s != "" {
|
|
return s
|
|
}
|
|
return "[" + key + "]"
|
|
})
|
|
}
|
|
|
|
type MatrixData struct {
|
|
Legend []string `json:"legend"`
|
|
Series []MatrixLine `json:"series"`
|
|
}
|
|
|
|
type MatrixLine struct {
|
|
Name string `json:"name"`
|
|
Data []MatrixPoint `json:"data"`
|
|
}
|
|
|
|
type MatrixPoint struct {
|
|
X int64 `json:"x"`
|
|
Y float64 `json:"y"`
|
|
}
|
|
|
|
func (p *MatrixPoint) MarshalJSON() ([]byte, error) {
|
|
return cast.StringToBytes(fmt.Sprintf("[%v,%v]", p.X, p.Y)), nil
|
|
}
|
|
|
|
type VectorData struct {
|
|
Legend []string `json:"legend"`
|
|
Data []VectorValue `json:"data"`
|
|
}
|
|
|
|
type VectorValue struct {
|
|
Name string `json:"name"`
|
|
Value float64 `json:"value"`
|
|
}
|