You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
106 lines
2.6 KiB
106 lines
2.6 KiB
package metrics |
|
|
|
import ( |
|
"context" |
|
"strconv" |
|
"time" |
|
|
|
"github.com/go-kratos/kratos/v2/errors" |
|
"github.com/go-kratos/kratos/v2/metrics" |
|
"github.com/go-kratos/kratos/v2/middleware" |
|
"github.com/go-kratos/kratos/v2/transport" |
|
) |
|
|
|
// Option is metrics option. |
|
type Option func(*options) |
|
|
|
// WithRequests with requests counter. |
|
func WithRequests(c metrics.Counter) Option { |
|
return func(o *options) { |
|
o.requests = c |
|
} |
|
} |
|
|
|
// WithSeconds with seconds histogram. |
|
func WithSeconds(c metrics.Observer) Option { |
|
return func(o *options) { |
|
o.seconds = c |
|
} |
|
} |
|
|
|
type options struct { |
|
// counter: <client/server>_requests_code_total{kind, operation, code, reason} |
|
requests metrics.Counter |
|
// histogram: <client/server>_requests_seconds_bucket{kind, operation} |
|
seconds metrics.Observer |
|
} |
|
|
|
// Server is middleware server-side metrics. |
|
func Server(opts ...Option) middleware.Middleware { |
|
op := options{} |
|
for _, o := range opts { |
|
o(&op) |
|
} |
|
return func(handler middleware.Handler) middleware.Handler { |
|
return func(ctx context.Context, req interface{}) (interface{}, error) { |
|
var ( |
|
code int |
|
reason string |
|
kind string |
|
operation string |
|
) |
|
startTime := time.Now() |
|
if info, ok := transport.FromServerContext(ctx); ok { |
|
kind = info.Kind().String() |
|
operation = info.Operation() |
|
} |
|
reply, err := handler(ctx, req) |
|
if se := errors.FromError(err); se != nil { |
|
code = int(se.Code) |
|
reason = se.Reason |
|
} |
|
if op.requests != nil { |
|
op.requests.With(kind, operation, strconv.Itoa(code), reason).Inc() |
|
} |
|
if op.seconds != nil { |
|
op.seconds.With(kind, operation).Observe(time.Since(startTime).Seconds()) |
|
} |
|
return reply, err |
|
} |
|
} |
|
} |
|
|
|
// Client is middleware client-side metrics. |
|
func Client(opts ...Option) middleware.Middleware { |
|
op := options{} |
|
for _, o := range opts { |
|
o(&op) |
|
} |
|
return func(handler middleware.Handler) middleware.Handler { |
|
return func(ctx context.Context, req interface{}) (interface{}, error) { |
|
var ( |
|
code int |
|
reason string |
|
kind string |
|
operation string |
|
) |
|
startTime := time.Now() |
|
if info, ok := transport.FromClientContext(ctx); ok { |
|
kind = info.Kind().String() |
|
operation = info.Operation() |
|
} |
|
reply, err := handler(ctx, req) |
|
if se := errors.FromError(err); se != nil { |
|
code = int(se.Code) |
|
reason = se.Reason |
|
} |
|
if op.requests != nil { |
|
op.requests.With(kind, operation, strconv.Itoa(code), reason).Inc() |
|
} |
|
if op.seconds != nil { |
|
op.seconds.With(kind, operation).Observe(time.Since(startTime).Seconds()) |
|
} |
|
return reply, err |
|
} |
|
} |
|
}
|
|
|