-
Notifications
You must be signed in to change notification settings - Fork 33
/
Copy pathwalgen.go
127 lines (107 loc) · 3.6 KB
/
walgen.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
package walgen
import (
"context"
"math/rand"
"os"
"runtime"
"strconv"
"strings"
"time"
"github.com/go-kit/log"
"github.com/go-kit/log/level"
"github.com/pkg/errors"
"github.com/prometheus/common/model"
"github.com/prometheus/prometheus/model/labels"
"github.com/prometheus/prometheus/model/timestamp"
"github.com/prometheus/prometheus/storage"
"github.com/prometheus/prometheus/tsdb"
"github.com/thanos-io/thanosbench/pkg/seriesgen"
)
// TODO(bwplotka): Allow more realistic output.
type Config struct {
InputSeries []Series
Retention time.Duration
ScrapeInterval time.Duration
}
type Series struct {
Type string // gauge, counter (if counter we treat below as rate aim).
Characteristics seriesgen.Characteristics
// Result is an exact Prometheus HTTP query result that would be used to generate series' metrics labels.
Result QueryData
// Replicate multiples this set given number of times. For example if result has 10 metrics and replicate is 10 we will
// have 100 unique series.
Replicate int
}
type QueryData struct {
ResultType model.ValueType `json:"resultType"`
Result model.Vector `json:"result"`
}
func GenerateTSDBWAL(logger log.Logger, dir string, config Config) error {
if config.ScrapeInterval == 0 {
config.ScrapeInterval = 15 * time.Second
}
maxBlockDuration := config.Retention / 10
// TODO(bwplotka): Moved to something like https://github.com/thanos-io/thanos/blob/master/pkg/testutil/prometheus.go#L289
// to actually generate blocks! It will be fine for TSDB use cases as well.
db, err := tsdb.Open(dir, nil, nil, &tsdb.Options{
MinBlockDuration: int64(2 * time.Hour / time.Millisecond),
MaxBlockDuration: maxBlockDuration.Milliseconds(),
RetentionDuration: config.Retention.Milliseconds(),
NoLockfile: true,
}, nil)
if err != nil {
level.Error(logger).Log("err", err)
os.Exit(1)
}
// Of course there will be small gap in minTime vs time.Now once we finish.
// We are fine with this.
n := time.Now()
maxTime := timestamp.FromTime(n)
minTime := timestamp.FromTime(n.Add(-config.Retention))
random := rand.New(rand.NewSource(1234))
set := &Set{}
for _, in := range config.InputSeries {
for _, r := range in.Result.Result {
for i := 0; i < in.Replicate; i++ {
lset := labels.New()
for n, v := range r.Metric {
lset = append(lset, labels.Label{Name: string(n), Value: string(v)})
}
if i > 0 {
lset = append(lset, labels.Label{Name: "blockgen_fake_replica", Value: strconv.Itoa(i)})
}
switch strings.ToLower(in.Type) {
case "counter":
set.s = append(set.s, seriesgen.NewSeriesGen(lset, seriesgen.NewCounterGen(random, minTime, maxTime, in.Characteristics)))
case "gauge":
set.s = append(set.s, seriesgen.NewSeriesGen(lset, seriesgen.NewGaugeGen(random, minTime, maxTime, in.Characteristics)))
default:
return errors.Errorf("failed to parse series, unknown metric type: %s", in.Type)
}
}
}
}
if err := seriesgen.Append(context.Background(), 2*runtime.GOMAXPROCS(0), db, set); err != nil {
return errors.Wrap(err, "commit")
}
// Don't wait for compact, it will be compacted by Prometheus anyway.
if err := db.Close(); err != nil {
return errors.Wrap(err, "close")
}
level.Info(logger).Log("msg", "generated artificial metrics", "series", len(set.s))
return nil
}
type Set struct {
s []storage.Series
curr int
}
func (s *Set) Next() bool {
if s.curr >= len(s.s) {
return false
}
s.curr++
return true
}
func (s *Set) At() storage.Series { return s.s[s.curr-1] }
func (s *Set) Err() error { return nil }
func (s *Set) Warnings() storage.Warnings { return storage.Warnings{} }