This repository was archived by the owner on Aug 23, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 107
/
Copy pathgraphite.go
242 lines (204 loc) · 6.65 KB
/
graphite.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
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
package models
import (
"bytes"
"fmt"
"sort"
"strconv"
"github.com/go-macaron/binding"
"github.com/grafana/metrictank/idx"
pickle "github.com/kisielk/og-rek"
opentracing "github.com/opentracing/opentracing-go"
traceLog "github.com/opentracing/opentracing-go/log"
"gopkg.in/macaron.v1"
)
//go:generate msgp
//msgp:ignore FromTo
//msgp:ignore GraphiteAutoCompleteTags
//msgp:ignore GraphiteAutoCompleteTagValues
//msgp:ignore GraphiteFind
//msgp:ignore GraphiteRender
//msgp:ignore GraphiteTag
//msgp:ignore GraphiteTagDetails
//msgp:ignore GraphiteTagDetailsResp
//msgp:ignore GraphiteTagDetailsValueResp
//msgp:ignore GraphiteTagFindSeries
//msgp:ignore GraphiteTagFindSeriesResp
//msgp:ignore GraphiteTagResp
//msgp:ignore GraphiteTags
//msgp:ignore GraphiteTagsResp
//msgp:ignore MetricNames
//msgp:ignore MetricsDelete
//msgp:ignore SeriesCompleter
//msgp:ignore SeriesCompleterItem
//msgp:ignore SeriesTree
//msgp:ignore SeriesTreeItem
type FromTo struct {
From string `json:"from" form:"from"`
Until string `json:"until" form:"until"`
To string `json:"to" form:"to"` // graphite uses 'until' but we allow to alternatively cause it's shorter
Tz string `json:"tz" form:"tz"`
}
type GraphiteRender struct {
FromTo
MaxDataPoints uint32 `json:"maxDataPoints" form:"maxDataPoints" binding:"Default(800)"`
Targets []string `json:"target" form:"target"`
TargetsRails []string `form:"target[]"` // # Rails/PHP/jQuery common practice format: ?target[]=path.1&target[]=path.2 -> like graphite, we allow this.
Format string `json:"format" form:"format" binding:"In(,json,msgp,msgpack,pickle)"`
NoProxy bool `json:"local" form:"local"` //this is set to true by graphite-web when it passes request to cluster servers
Meta bool `json:"meta" form:"meta"` // request for meta data, which will be returned as long as the format is compatible (json) and we don't have to go via graphite
Process string `json:"process" form:"process" binding:"In(,none,stable,any);Default(stable)"`
}
func (gr GraphiteRender) Validate(ctx *macaron.Context, errs binding.Errors) binding.Errors {
if len(gr.Targets) == 0 {
if len(gr.TargetsRails) == 0 {
errs = append(errs, binding.Error{
FieldNames: []string{"target"},
Classification: "RequiredError",
Message: "Required",
})
return errs
}
gr.Targets = gr.TargetsRails
}
for _, val := range gr.Targets {
if val == "" {
errs = append(errs, binding.Error{
FieldNames: []string{"target"},
Classification: "RequiredError",
Message: "Required",
})
}
}
return errs
}
type GraphiteTags struct {
Filter string `json:"filter" form:"filter"`
}
type GraphiteTagsResp []GraphiteTagResp
type GraphiteAutoCompleteTags struct {
Prefix string `json:"tagPrefix" form:"tagPrefix"`
Expr []string `json:"expr" form:"expr"`
Limit uint `json:"limit" form:"limit"`
}
type GraphiteAutoCompleteTagValues struct {
Tag string `json:"tag" form:"tag"`
Prefix string `json:"valuePrefix" form:"valuePrefix"`
Expr []string `json:"expr" form:"expr"`
Limit uint `json:"limit" form:"limit"`
}
type GraphiteTagResp struct {
Tag string `json:"tag"`
}
type GraphiteTagDetails struct {
Tag string `json:"tag" form:"tag"`
Filter string `json:"filter" form:"filter"`
}
type GraphiteTagDetailsResp struct {
Tag string `json:"tag"`
Values []GraphiteTagDetailsValueResp `json:"values"`
}
type GraphiteTagDetailsValueResp struct {
Count uint64 `json:"count"`
Value string `json:"value"`
}
type GraphiteTagFindSeries struct {
Expr []string `json:"expr" form:"expr"`
From int64 `json:"from" form:"from"`
}
type GraphiteTagFindSeriesResp struct {
Series []string `json:"series"`
}
type GraphiteTagDelSeries struct {
Paths []string `json:"path" form:"path"`
Propagate bool `json:"propagate" form:"propagate" binding:"Default(true)"`
}
func (g GraphiteTagDelSeries) Trace(span opentracing.Span) {
span.LogFields(
traceLog.String("paths", fmt.Sprintf("%q", g.Paths)),
traceLog.Bool("propagate", g.Propagate),
)
}
func (g GraphiteTagDelSeries) TraceDebug(span opentracing.Span) {
}
type GraphiteTagDelSeriesResp struct {
Count int `json:"count"`
Peers map[string]int `json:"peers"`
}
type GraphiteFind struct {
FromTo
Query string `json:"query" form:"query" binding:"Required"`
Format string `json:"format" form:"format" binding:"In(,completer,json,treejson,msgpack,pickle)"`
Jsonp string `json:"jsonp" form:"jsonp"`
}
type MetricsDelete struct {
Query string `json:"query" form:"query" binding:"Required"`
}
type MetricNames []idx.Archive
func (defs MetricNames) MarshalJSONFast(b []byte) ([]byte, error) {
seen := make(map[string]struct{})
names := make([]string, 0, len(defs))
for i := 0; i < len(defs); i++ {
_, ok := seen[defs[i].Name]
if !ok {
names = append(names, defs[i].Name)
seen[defs[i].Name] = struct{}{}
}
}
sort.Strings(names)
b = append(b, '[')
for _, name := range names {
b = strconv.AppendQuoteToASCII(b, name)
b = append(b, ',')
}
if len(defs) != 0 {
b = b[:len(b)-1] // cut last comma
}
b = append(b, ']')
return b, nil
}
func (defs MetricNames) MarshalJSON() ([]byte, error) {
return defs.MarshalJSONFast(nil)
}
type SeriesCompleter map[string][]SeriesCompleterItem
func NewSeriesCompleter() SeriesCompleter {
return SeriesCompleter(map[string][]SeriesCompleterItem{"metrics": make([]SeriesCompleterItem, 0)})
}
func (c SeriesCompleter) Add(e SeriesCompleterItem) {
c["metrics"] = append(c["metrics"], e)
}
type SeriesCompleterItem struct {
Path string `json:"path"`
Name string `json:"name"`
IsLeaf string `json:"is_leaf"`
}
type SeriesPickle []SeriesPickleItem
func (s SeriesPickle) Pickle(buf []byte) ([]byte, error) {
buffer := bytes.NewBuffer(buf)
encoder := pickle.NewEncoder(buffer)
err := encoder.Encode(s)
return buffer.Bytes(), err
}
type SeriesPickleItem struct {
Path string `pickle:"path" msg:"path"`
IsLeaf bool `pickle:"isLeaf" msg:"isLeaf"`
Intervals [][]int64 `pickle:"intervals" msg:"intervals"` // list of (start,end) tuples
}
func NewSeriesPickleItem(path string, isLeaf bool, intervals [][]int64) SeriesPickleItem {
return SeriesPickleItem{
Path: path,
IsLeaf: isLeaf,
Intervals: intervals,
}
}
type SeriesTree []SeriesTreeItem
func (s *SeriesTree) Add(i *SeriesTreeItem) {
*s = append(*s, *i)
}
type SeriesTreeItem struct {
AllowChildren int `json:"allowChildren"`
Expandable int `json:"expandable"`
Leaf int `json:"leaf"`
ID string `json:"id"`
Text string `json:"text"`
Context map[string]int `json:"context"` // unused
}