-
Notifications
You must be signed in to change notification settings - Fork 106
/
metrics.go
102 lines (90 loc) · 3.6 KB
/
metrics.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
/*
Copyright 2017 Caicloud Authors
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package metrics
import (
"context"
"time"
"github.com/prometheus/client_golang/prometheus/promhttp"
"github.com/caicloud/nirvana/definition"
"github.com/caicloud/nirvana/metrics"
"github.com/caicloud/nirvana/service"
)
// Default returns a metric Middleware under the namespace "nirvana" for restful descriptor.
//
// Once called, the namespace is set and can not be changed. Future attempt to build more Middleware
// will result in ones with the same namespace as the first one.
//
// Unlike the metrics plugin which takes care of everything, you must call Descriptor() to build a
// Descriptor and configure it to a server yourself.
func Default() definition.Middleware {
return Restful(nil)
}
// Restful returns a metrics Middleware for Restful Descriptors built from the given options.
//
// Once called, the namespace is set and can not be changed. Future attempt to build more Middleware
// will result in ones with the same namespace as the first one.
//
// Unlike the metrics plugin which takes care of everything, you must call Descriptor() to build a
// Descriptor and configure it to a server yourself.
func Restful(options *metrics.Options) definition.Middleware {
metrics.Install(options)
return func(ctx context.Context, next definition.Chain) error {
startTime := time.Now()
err := next.Continue(ctx)
httpCtx := service.HTTPContextFrom(ctx)
resp := httpCtx.ResponseWriter()
metrics.RecordRestfulRequest(
httpCtx.RoutePath(), httpCtx.Request().Method,
resp.StatusCode(), resp.ContentLength(), time.Since(startTime),
)
return err
}
}
// RPC returns a metrics Middleware for RCP Descriptors built from the given options.
//
// Once called, the namespace is set and can not be changed. Future attempt to build more Middleware
// will result in ones with the same namespace as the first one.
//
// Unlike the metrics plugin which takes care of everything, you must call Descriptor() to build a
// Descriptor and configure it to a server yourself.
func RPC(options *metrics.Options) definition.Middleware {
metrics.Install(options)
return func(ctx context.Context, next definition.Chain) error {
startTime := time.Now()
err := next.Continue(ctx)
httpCtx := service.HTTPContextFrom(ctx)
resp := httpCtx.ResponseWriter()
metrics.RecordRPCRequest(
httpCtx.Request().URL.Query().Get("Action"),
httpCtx.Request().URL.Query().Get("Version"),
resp.StatusCode(), resp.ContentLength(), time.Since(startTime),
)
return err
}
}
// Descriptor returns a descriptor for the API; it must be configured to a server in order to serve the
// metric API.
func Descriptor(path string) definition.Descriptor {
if path == "" {
path = "/metrics"
}
return definition.SimpleDescriptor(definition.Get, path, service.WrapHTTPHandler(promhttp.Handler()))
}
// RPCDescriptor returns a rpc descriptor for the API; it must be configured to a server in order to serve the
// metric API.
func RPCDescriptor(path string) definition.RPCDescriptor {
if path == "" {
path = "/metrics"
}
return definition.SimpleRPCDescriptor(path, service.WrapHTTPHandler(promhttp.Handler()))
}