mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2024-12-31 15:06:26 +00:00
Merge remote-tracking branch 'origin/lts-1.93' into pmm-6401-read-prometheus-data-files
This commit is contained in:
commit
ca20478a69
30 changed files with 280 additions and 85 deletions
|
@ -173,7 +173,8 @@ VictoriaMetrics is developed at a fast pace, so it is recommended periodically c
|
||||||
|
|
||||||
### Environment variables
|
### Environment variables
|
||||||
|
|
||||||
All the VictoriaMetrics components allow referring environment variables in command-line flags via `%{ENV_VAR}` syntax.
|
All the VictoriaMetrics components allow referring environment variables in `yaml` configuration files (such as `-promscrape.config`)
|
||||||
|
and in command-line flags via `%{ENV_VAR}` syntax.
|
||||||
For example, `-metricsAuthKey=%{METRICS_AUTH_KEY}` is automatically expanded to `-metricsAuthKey=top-secret`
|
For example, `-metricsAuthKey=%{METRICS_AUTH_KEY}` is automatically expanded to `-metricsAuthKey=top-secret`
|
||||||
if `METRICS_AUTH_KEY=top-secret` environment variable exists at VictoriaMetrics startup.
|
if `METRICS_AUTH_KEY=top-secret` environment variable exists at VictoriaMetrics startup.
|
||||||
This expansion is performed by VictoriaMetrics itself.
|
This expansion is performed by VictoriaMetrics itself.
|
||||||
|
|
|
@ -87,8 +87,8 @@ func initLabelsGlobal() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (rctx *relabelCtx) applyRelabeling(tss []prompbmarshal.TimeSeries, extraLabels []prompbmarshal.Label, pcs *promrelabel.ParsedConfigs) []prompbmarshal.TimeSeries {
|
func (rctx *relabelCtx) applyRelabeling(tss []prompbmarshal.TimeSeries, pcs *promrelabel.ParsedConfigs) []prompbmarshal.TimeSeries {
|
||||||
if len(extraLabels) == 0 && pcs.Len() == 0 && !*usePromCompatibleNaming {
|
if pcs.Len() == 0 && !*usePromCompatibleNaming {
|
||||||
// Nothing to change.
|
// Nothing to change.
|
||||||
return tss
|
return tss
|
||||||
}
|
}
|
||||||
|
@ -98,34 +98,15 @@ func (rctx *relabelCtx) applyRelabeling(tss []prompbmarshal.TimeSeries, extraLab
|
||||||
ts := &tss[i]
|
ts := &tss[i]
|
||||||
labelsLen := len(labels)
|
labelsLen := len(labels)
|
||||||
labels = append(labels, ts.Labels...)
|
labels = append(labels, ts.Labels...)
|
||||||
// extraLabels must be added before applying relabeling according to https://prometheus.io/docs/prometheus/latest/configuration/configuration/#remote_write
|
|
||||||
for j := range extraLabels {
|
|
||||||
extraLabel := &extraLabels[j]
|
|
||||||
tmp := promrelabel.GetLabelByName(labels[labelsLen:], extraLabel.Name)
|
|
||||||
if tmp != nil {
|
|
||||||
tmp.Value = extraLabel.Value
|
|
||||||
} else {
|
|
||||||
labels = append(labels, *extraLabel)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if *usePromCompatibleNaming {
|
|
||||||
// Replace unsupported Prometheus chars in label names and metric names with underscores.
|
|
||||||
tmpLabels := labels[labelsLen:]
|
|
||||||
for j := range tmpLabels {
|
|
||||||
label := &tmpLabels[j]
|
|
||||||
if label.Name == "__name__" {
|
|
||||||
label.Value = promrelabel.SanitizeName(label.Value)
|
|
||||||
} else {
|
|
||||||
label.Name = promrelabel.SanitizeName(label.Name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
labels = pcs.Apply(labels, labelsLen)
|
labels = pcs.Apply(labels, labelsLen)
|
||||||
labels = promrelabel.FinalizeLabels(labels[:labelsLen], labels[labelsLen:])
|
labels = promrelabel.FinalizeLabels(labels[:labelsLen], labels[labelsLen:])
|
||||||
if len(labels) == labelsLen {
|
if len(labels) == labelsLen {
|
||||||
// Drop the current time series, since relabeling removed all the labels.
|
// Drop the current time series, since relabeling removed all the labels.
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
if *usePromCompatibleNaming {
|
||||||
|
fixPromCompatibleNaming(labels[labelsLen:])
|
||||||
|
}
|
||||||
tssDst = append(tssDst, prompbmarshal.TimeSeries{
|
tssDst = append(tssDst, prompbmarshal.TimeSeries{
|
||||||
Labels: labels[labelsLen:],
|
Labels: labels[labelsLen:],
|
||||||
Samples: ts.Samples,
|
Samples: ts.Samples,
|
||||||
|
@ -135,6 +116,32 @@ func (rctx *relabelCtx) applyRelabeling(tss []prompbmarshal.TimeSeries, extraLab
|
||||||
return tssDst
|
return tssDst
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (rctx *relabelCtx) appendExtraLabels(tss []prompbmarshal.TimeSeries, extraLabels []prompbmarshal.Label) {
|
||||||
|
if len(extraLabels) == 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
labels := rctx.labels[:0]
|
||||||
|
for i := range tss {
|
||||||
|
ts := &tss[i]
|
||||||
|
labelsLen := len(labels)
|
||||||
|
labels = append(labels, ts.Labels...)
|
||||||
|
for j := range extraLabels {
|
||||||
|
extraLabel := extraLabels[j]
|
||||||
|
if *usePromCompatibleNaming {
|
||||||
|
extraLabel.Name = promrelabel.SanitizeLabelName(extraLabel.Name)
|
||||||
|
}
|
||||||
|
tmp := promrelabel.GetLabelByName(labels[labelsLen:], extraLabel.Name)
|
||||||
|
if tmp != nil {
|
||||||
|
tmp.Value = extraLabel.Value
|
||||||
|
} else {
|
||||||
|
labels = append(labels, extraLabel)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ts.Labels = labels[labelsLen:]
|
||||||
|
}
|
||||||
|
rctx.labels = labels
|
||||||
|
}
|
||||||
|
|
||||||
type relabelCtx struct {
|
type relabelCtx struct {
|
||||||
// pool for labels, which are used during the relabeling.
|
// pool for labels, which are used during the relabeling.
|
||||||
labels []prompbmarshal.Label
|
labels []prompbmarshal.Label
|
||||||
|
@ -159,3 +166,15 @@ func putRelabelCtx(rctx *relabelCtx) {
|
||||||
rctx.labels = rctx.labels[:0]
|
rctx.labels = rctx.labels[:0]
|
||||||
relabelCtxPool.Put(rctx)
|
relabelCtxPool.Put(rctx)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func fixPromCompatibleNaming(labels []prompbmarshal.Label) {
|
||||||
|
// Replace unsupported Prometheus chars in label names and metric names with underscores.
|
||||||
|
for i := range labels {
|
||||||
|
label := &labels[i]
|
||||||
|
if label.Name == "__name__" {
|
||||||
|
label.Value = promrelabel.SanitizeMetricName(label.Value)
|
||||||
|
} else {
|
||||||
|
label.Name = promrelabel.SanitizeLabelName(label.Name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -10,18 +10,16 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestApplyRelabeling(t *testing.T) {
|
func TestApplyRelabeling(t *testing.T) {
|
||||||
f := func(extraLabels []prompbmarshal.Label, pcs *promrelabel.ParsedConfigs, sTss, sExpTss string) {
|
f := func(pcs *promrelabel.ParsedConfigs, sTss, sExpTss string) {
|
||||||
rctx := &relabelCtx{}
|
rctx := &relabelCtx{}
|
||||||
tss, expTss := parseSeries(sTss), parseSeries(sExpTss)
|
tss, expTss := parseSeries(sTss), parseSeries(sExpTss)
|
||||||
gotTss := rctx.applyRelabeling(tss, extraLabels, pcs)
|
gotTss := rctx.applyRelabeling(tss, pcs)
|
||||||
if !reflect.DeepEqual(gotTss, expTss) {
|
if !reflect.DeepEqual(gotTss, expTss) {
|
||||||
t.Fatalf("expected to have: \n%v;\ngot: \n%v", expTss, gotTss)
|
t.Fatalf("expected to have: \n%v;\ngot: \n%v", expTss, gotTss)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
f(nil, nil, "up", "up")
|
f(nil, "up", "up")
|
||||||
f([]prompbmarshal.Label{{Name: "foo", Value: "bar"}}, nil, "up", `up{foo="bar"}`)
|
|
||||||
f([]prompbmarshal.Label{{Name: "foo", Value: "bar"}}, nil, `up{foo="baz"}`, `up{foo="bar"}`)
|
|
||||||
|
|
||||||
pcs, err := promrelabel.ParseRelabelConfigsData([]byte(`
|
pcs, err := promrelabel.ParseRelabelConfigsData([]byte(`
|
||||||
- target_label: "foo"
|
- target_label: "foo"
|
||||||
|
@ -32,11 +30,32 @@ func TestApplyRelabeling(t *testing.T) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("unexpected error: %s", err)
|
t.Fatalf("unexpected error: %s", err)
|
||||||
}
|
}
|
||||||
f(nil, pcs, `up{foo="baz", env="prod"}`, `up{foo="aaa"}`)
|
f(pcs, `up{foo="baz", env="prod"}`, `up{foo="aaa"}`)
|
||||||
|
|
||||||
oldVal := *usePromCompatibleNaming
|
oldVal := *usePromCompatibleNaming
|
||||||
*usePromCompatibleNaming = true
|
*usePromCompatibleNaming = true
|
||||||
f(nil, nil, `foo.bar`, `foo_bar`)
|
f(nil, `foo.bar`, `foo_bar`)
|
||||||
|
*usePromCompatibleNaming = oldVal
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAppendExtraLabels(t *testing.T) {
|
||||||
|
f := func(extraLabels []prompbmarshal.Label, sTss, sExpTss string) {
|
||||||
|
rctx := &relabelCtx{}
|
||||||
|
tss, expTss := parseSeries(sTss), parseSeries(sExpTss)
|
||||||
|
rctx.appendExtraLabels(tss, extraLabels)
|
||||||
|
if !reflect.DeepEqual(tss, expTss) {
|
||||||
|
t.Fatalf("expected to have: \n%v;\ngot: \n%v", expTss, tss)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
f(nil, "up", "up")
|
||||||
|
f([]prompbmarshal.Label{{Name: "foo", Value: "bar"}}, "up", `up{foo="bar"}`)
|
||||||
|
f([]prompbmarshal.Label{{Name: "foo", Value: "bar"}}, `up{foo="baz"}`, `up{foo="bar"}`)
|
||||||
|
f([]prompbmarshal.Label{{Name: "baz", Value: "qux"}}, `up{foo="baz"}`, `up{foo="baz",baz="qux"}`)
|
||||||
|
|
||||||
|
oldVal := *usePromCompatibleNaming
|
||||||
|
*usePromCompatibleNaming = true
|
||||||
|
f([]prompbmarshal.Label{{Name: "foo.bar", Value: "baz"}}, "up", `up{foo_bar="baz"}`)
|
||||||
*usePromCompatibleNaming = oldVal
|
*usePromCompatibleNaming = oldVal
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -355,7 +355,7 @@ func Push(at *auth.Token, wr *prompbmarshal.WriteRequest) {
|
||||||
var rctx *relabelCtx
|
var rctx *relabelCtx
|
||||||
rcs := allRelabelConfigs.Load()
|
rcs := allRelabelConfigs.Load()
|
||||||
pcsGlobal := rcs.global
|
pcsGlobal := rcs.global
|
||||||
if pcsGlobal.Len() > 0 || len(labelsGlobal) > 0 {
|
if pcsGlobal.Len() > 0 {
|
||||||
rctx = getRelabelCtx()
|
rctx = getRelabelCtx()
|
||||||
}
|
}
|
||||||
tss := wr.Timeseries
|
tss := wr.Timeseries
|
||||||
|
@ -386,7 +386,7 @@ func Push(at *auth.Token, wr *prompbmarshal.WriteRequest) {
|
||||||
}
|
}
|
||||||
if rctx != nil {
|
if rctx != nil {
|
||||||
rowsCountBeforeRelabel := getRowsCount(tssBlock)
|
rowsCountBeforeRelabel := getRowsCount(tssBlock)
|
||||||
tssBlock = rctx.applyRelabeling(tssBlock, labelsGlobal, pcsGlobal)
|
tssBlock = rctx.applyRelabeling(tssBlock, pcsGlobal)
|
||||||
rowsCountAfterRelabel := getRowsCount(tssBlock)
|
rowsCountAfterRelabel := getRowsCount(tssBlock)
|
||||||
rowsDroppedByGlobalRelabel.Add(rowsCountBeforeRelabel - rowsCountAfterRelabel)
|
rowsDroppedByGlobalRelabel.Add(rowsCountBeforeRelabel - rowsCountAfterRelabel)
|
||||||
}
|
}
|
||||||
|
@ -668,7 +668,7 @@ func (rwctx *remoteWriteCtx) Push(tss []prompbmarshal.TimeSeries) {
|
||||||
v = tssPool.Get().(*[]prompbmarshal.TimeSeries)
|
v = tssPool.Get().(*[]prompbmarshal.TimeSeries)
|
||||||
tss = append(*v, tss...)
|
tss = append(*v, tss...)
|
||||||
rowsCountBeforeRelabel := getRowsCount(tss)
|
rowsCountBeforeRelabel := getRowsCount(tss)
|
||||||
tss = rctx.applyRelabeling(tss, nil, pcs)
|
tss = rctx.applyRelabeling(tss, pcs)
|
||||||
rowsCountAfterRelabel := getRowsCount(tss)
|
rowsCountAfterRelabel := getRowsCount(tss)
|
||||||
rwctx.rowsDroppedByRelabel.Add(rowsCountBeforeRelabel - rowsCountAfterRelabel)
|
rwctx.rowsDroppedByRelabel.Add(rowsCountBeforeRelabel - rowsCountAfterRelabel)
|
||||||
}
|
}
|
||||||
|
@ -719,6 +719,12 @@ func dropAggregatedSeries(src []prompbmarshal.TimeSeries, matchIdxs []byte, drop
|
||||||
}
|
}
|
||||||
|
|
||||||
func (rwctx *remoteWriteCtx) pushInternal(tss []prompbmarshal.TimeSeries) {
|
func (rwctx *remoteWriteCtx) pushInternal(tss []prompbmarshal.TimeSeries) {
|
||||||
|
if len(labelsGlobal) > 0 {
|
||||||
|
rctx := getRelabelCtx()
|
||||||
|
defer putRelabelCtx(rctx)
|
||||||
|
rctx.appendExtraLabels(tss, labelsGlobal)
|
||||||
|
}
|
||||||
|
|
||||||
pss := rwctx.pss
|
pss := rwctx.pss
|
||||||
idx := atomic.AddUint64(&rwctx.pssNextIdx, 1) % uint64(len(pss))
|
idx := atomic.AddUint64(&rwctx.pssNextIdx, 1) % uint64(len(pss))
|
||||||
pss[idx].Push(tss)
|
pss[idx].Push(tss)
|
||||||
|
|
|
@ -205,7 +205,20 @@ func hasFilepathPrefix(path, prefix string) bool {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
return strings.HasPrefix(pathAbs, prefixAbs)
|
if prefixAbs == pathAbs {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
rel, err := filepath.Rel(prefixAbs, pathAbs)
|
||||||
|
if err != nil {
|
||||||
|
// if paths can't be related - they don't match
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if i := strings.Index(rel, "."); i == 0 {
|
||||||
|
// if path can be related only with . as first char - they still don't match
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
// if paths are related - it is a match
|
||||||
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
func newOriginFS() (common.OriginFS, error) {
|
func newOriginFS() (common.OriginFS, error) {
|
||||||
|
|
|
@ -26,4 +26,9 @@ func TestHasFilepathPrefix(t *testing.T) {
|
||||||
f("fs://"+pwd+"/foo", pwd+"/foo/bar", false)
|
f("fs://"+pwd+"/foo", pwd+"/foo/bar", false)
|
||||||
f("fs://"+pwd+"/foo/bar", pwd+"/foo", true)
|
f("fs://"+pwd+"/foo/bar", pwd+"/foo", true)
|
||||||
f("fs://"+pwd+"/foo", pwd+"/bar", false)
|
f("fs://"+pwd+"/foo", pwd+"/bar", false)
|
||||||
|
f("fs:///data1", "/data", false)
|
||||||
|
f("fs:///data", "/data1", false)
|
||||||
|
f("fs:///data", "/data/foo", false)
|
||||||
|
f("fs:///data/foo", "/data", true)
|
||||||
|
f("fs:///data/foo/", "/data/", true)
|
||||||
}
|
}
|
||||||
|
|
|
@ -826,7 +826,7 @@ It is possible split migration process into set of smaller batches based on time
|
||||||
migrating large volumes of data as this adds indication of progress and ability to restore process from certain point
|
migrating large volumes of data as this adds indication of progress and ability to restore process from certain point
|
||||||
in case of failure.
|
in case of failure.
|
||||||
|
|
||||||
To use this you need to specify `--vm-native-step-interval` flag. Supported values are: `month`, `day`, `hour`, `minute`.
|
To use this you need to specify `--vm-native-step-interval` flag. Supported values are: `month`, `week`, `day`, `hour`, `minute`.
|
||||||
Note that in order to use this it is required `--vm-native-filter-time-start` to be set to calculate time ranges for
|
Note that in order to use this it is required `--vm-native-filter-time-start` to be set to calculate time ranges for
|
||||||
export process.
|
export process.
|
||||||
|
|
||||||
|
@ -836,7 +836,7 @@ Every range is being processed independently, which means that:
|
||||||
so it is possible to restart process starting from failed range.
|
so it is possible to restart process starting from failed range.
|
||||||
|
|
||||||
It is recommended using the `month` step when migrating the data over multiple months,
|
It is recommended using the `month` step when migrating the data over multiple months,
|
||||||
since the migration with `day` and `hour` steps may take longer time to complete because of additional overhead.
|
since the migration with `week`, `day` and `hour` steps may take longer time to complete because of additional overhead.
|
||||||
|
|
||||||
Usage example:
|
Usage example:
|
||||||
```console
|
```console
|
||||||
|
|
|
@ -362,7 +362,8 @@ var (
|
||||||
},
|
},
|
||||||
&cli.StringFlag{
|
&cli.StringFlag{
|
||||||
Name: vmNativeStepInterval,
|
Name: vmNativeStepInterval,
|
||||||
Usage: fmt.Sprintf("Split export data into chunks. Requires setting --%s. Valid values are '%s','%s','%s','%s'.", vmNativeFilterTimeStart, stepper.StepMonth, stepper.StepDay, stepper.StepHour, stepper.StepMinute),
|
Usage: fmt.Sprintf("Split export data into chunks. Requires setting --%s. Valid values are '%s','%s','%s','%s','%s'.", vmNativeFilterTimeStart,
|
||||||
|
stepper.StepMonth, stepper.StepWeek, stepper.StepDay, stepper.StepHour, stepper.StepMinute),
|
||||||
Value: stepper.StepMonth,
|
Value: stepper.StepMonth,
|
||||||
},
|
},
|
||||||
&cli.BoolFlag{
|
&cli.BoolFlag{
|
||||||
|
|
|
@ -180,7 +180,7 @@ func modifyData(msg Metric, normalize bool) (Metric, error) {
|
||||||
/*
|
/*
|
||||||
replace bad characters in metric name with _ per the data model
|
replace bad characters in metric name with _ per the data model
|
||||||
*/
|
*/
|
||||||
finalMsg.Metric = promrelabel.SanitizeName(name)
|
finalMsg.Metric = promrelabel.SanitizeMetricName(name)
|
||||||
// replace bad characters in tag keys with _ per the data model
|
// replace bad characters in tag keys with _ per the data model
|
||||||
for key, value := range msg.Tags {
|
for key, value := range msg.Tags {
|
||||||
// if normalization requested, lowercase the key and value
|
// if normalization requested, lowercase the key and value
|
||||||
|
@ -191,7 +191,7 @@ func modifyData(msg Metric, normalize bool) (Metric, error) {
|
||||||
/*
|
/*
|
||||||
replace all explicitly bad characters with _
|
replace all explicitly bad characters with _
|
||||||
*/
|
*/
|
||||||
key = promrelabel.SanitizeName(key)
|
key = promrelabel.SanitizeLabelName(key)
|
||||||
// tags that start with __ are considered custom stats for internal prometheus stuff, we should drop them
|
// tags that start with __ are considered custom stats for internal prometheus stuff, we should drop them
|
||||||
if !strings.HasPrefix(key, "__") {
|
if !strings.HasPrefix(key, "__") {
|
||||||
finalMsg.Tags[key] = value
|
finalMsg.Tags[key] = value
|
||||||
|
|
|
@ -8,10 +8,10 @@ import (
|
||||||
const (
|
const (
|
||||||
// StepMonth represents a one month interval
|
// StepMonth represents a one month interval
|
||||||
StepMonth string = "month"
|
StepMonth string = "month"
|
||||||
// StepDay represents a one day interval
|
|
||||||
StepDay string = "day"
|
|
||||||
// StepWeek represents a one week interval
|
// StepWeek represents a one week interval
|
||||||
StepWeek string = "week"
|
StepWeek string = "week"
|
||||||
|
// StepDay represents a one day interval
|
||||||
|
StepDay string = "day"
|
||||||
// StepHour represents a one hour interval
|
// StepHour represents a one hour interval
|
||||||
StepHour string = "hour"
|
StepHour string = "hour"
|
||||||
// StepMinute represents a one minute interval
|
// StepMinute represents a one minute interval
|
||||||
|
|
|
@ -201,7 +201,12 @@ func (p *vmNativeProcessor) runBackfilling(ctx context.Context, tenantID string,
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(metrics) == 0 {
|
if len(metrics) == 0 {
|
||||||
return fmt.Errorf("no metrics found")
|
errMsg := "no metrics found"
|
||||||
|
if tenantID != "" {
|
||||||
|
errMsg = fmt.Sprintf("%s for tenant id: %s", errMsg, tenantID)
|
||||||
|
}
|
||||||
|
log.Println(errMsg)
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
foundSeriesMsg = fmt.Sprintf("Found %d metrics to import", len(metrics))
|
foundSeriesMsg = fmt.Sprintf("Found %d metrics to import", len(metrics))
|
||||||
}
|
}
|
||||||
|
|
|
@ -134,9 +134,9 @@ func (ctx *Ctx) ApplyRelabeling(labels []prompb.Label) []prompb.Label {
|
||||||
for i := range tmpLabels {
|
for i := range tmpLabels {
|
||||||
label := &tmpLabels[i]
|
label := &tmpLabels[i]
|
||||||
if label.Name == "__name__" {
|
if label.Name == "__name__" {
|
||||||
label.Value = promrelabel.SanitizeName(label.Value)
|
label.Value = promrelabel.SanitizeMetricName(label.Value)
|
||||||
} else {
|
} else {
|
||||||
label.Name = promrelabel.SanitizeName(label.Name)
|
label.Name = promrelabel.SanitizeLabelName(label.Name)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -78,7 +78,6 @@ publish-via-docker: \
|
||||||
--build-arg root_image=$(ROOT_IMAGE) \
|
--build-arg root_image=$(ROOT_IMAGE) \
|
||||||
--build-arg APP_NAME=$(APP_NAME) \
|
--build-arg APP_NAME=$(APP_NAME) \
|
||||||
--tag $(DOCKER_NAMESPACE)/$(APP_NAME):$(PKG_TAG)$(RACE) \
|
--tag $(DOCKER_NAMESPACE)/$(APP_NAME):$(PKG_TAG)$(RACE) \
|
||||||
--tag $(DOCKER_NAMESPACE)/$(APP_NAME):$(LATEST_TAG)$(RACE) \
|
|
||||||
-o type=image \
|
-o type=image \
|
||||||
-f app/$(APP_NAME)/multiarch/Dockerfile \
|
-f app/$(APP_NAME)/multiarch/Dockerfile \
|
||||||
--push \
|
--push \
|
||||||
|
|
|
@ -2,7 +2,7 @@ version: '3.5'
|
||||||
services:
|
services:
|
||||||
vmagent:
|
vmagent:
|
||||||
container_name: vmagent
|
container_name: vmagent
|
||||||
image: victoriametrics/vmagent:v1.92.1
|
image: victoriametrics/vmagent:v1.93.0
|
||||||
depends_on:
|
depends_on:
|
||||||
- "vminsert"
|
- "vminsert"
|
||||||
ports:
|
ports:
|
||||||
|
@ -32,7 +32,7 @@ services:
|
||||||
|
|
||||||
vmstorage-1:
|
vmstorage-1:
|
||||||
container_name: vmstorage-1
|
container_name: vmstorage-1
|
||||||
image: victoriametrics/vmstorage:v1.92.1-cluster
|
image: victoriametrics/vmstorage:v1.93.0-cluster
|
||||||
ports:
|
ports:
|
||||||
- 8482
|
- 8482
|
||||||
- 8400
|
- 8400
|
||||||
|
@ -44,7 +44,7 @@ services:
|
||||||
restart: always
|
restart: always
|
||||||
vmstorage-2:
|
vmstorage-2:
|
||||||
container_name: vmstorage-2
|
container_name: vmstorage-2
|
||||||
image: victoriametrics/vmstorage:v1.92.1-cluster
|
image: victoriametrics/vmstorage:v1.93.0-cluster
|
||||||
ports:
|
ports:
|
||||||
- 8482
|
- 8482
|
||||||
- 8400
|
- 8400
|
||||||
|
@ -56,7 +56,7 @@ services:
|
||||||
restart: always
|
restart: always
|
||||||
vminsert:
|
vminsert:
|
||||||
container_name: vminsert
|
container_name: vminsert
|
||||||
image: victoriametrics/vminsert:v1.92.1-cluster
|
image: victoriametrics/vminsert:v1.93.0-cluster
|
||||||
depends_on:
|
depends_on:
|
||||||
- "vmstorage-1"
|
- "vmstorage-1"
|
||||||
- "vmstorage-2"
|
- "vmstorage-2"
|
||||||
|
@ -68,7 +68,7 @@ services:
|
||||||
restart: always
|
restart: always
|
||||||
vmselect:
|
vmselect:
|
||||||
container_name: vmselect
|
container_name: vmselect
|
||||||
image: victoriametrics/vmselect:v1.92.1-cluster
|
image: victoriametrics/vmselect:v1.93.0-cluster
|
||||||
depends_on:
|
depends_on:
|
||||||
- "vmstorage-1"
|
- "vmstorage-1"
|
||||||
- "vmstorage-2"
|
- "vmstorage-2"
|
||||||
|
@ -82,7 +82,7 @@ services:
|
||||||
|
|
||||||
vmalert:
|
vmalert:
|
||||||
container_name: vmalert
|
container_name: vmalert
|
||||||
image: victoriametrics/vmalert:v1.92.1
|
image: victoriametrics/vmalert:v1.93.0
|
||||||
depends_on:
|
depends_on:
|
||||||
- "vmselect"
|
- "vmselect"
|
||||||
ports:
|
ports:
|
||||||
|
|
|
@ -2,7 +2,7 @@ version: "3.5"
|
||||||
services:
|
services:
|
||||||
vmagent:
|
vmagent:
|
||||||
container_name: vmagent
|
container_name: vmagent
|
||||||
image: victoriametrics/vmagent:v1.92.1
|
image: victoriametrics/vmagent:v1.93.0
|
||||||
depends_on:
|
depends_on:
|
||||||
- "victoriametrics"
|
- "victoriametrics"
|
||||||
ports:
|
ports:
|
||||||
|
@ -18,7 +18,7 @@ services:
|
||||||
restart: always
|
restart: always
|
||||||
victoriametrics:
|
victoriametrics:
|
||||||
container_name: victoriametrics
|
container_name: victoriametrics
|
||||||
image: victoriametrics/victoria-metrics:v1.92.1
|
image: victoriametrics/victoria-metrics:v1.93.0
|
||||||
ports:
|
ports:
|
||||||
- 8428:8428
|
- 8428:8428
|
||||||
- 8089:8089
|
- 8089:8089
|
||||||
|
@ -56,7 +56,7 @@ services:
|
||||||
restart: always
|
restart: always
|
||||||
vmalert:
|
vmalert:
|
||||||
container_name: vmalert
|
container_name: vmalert
|
||||||
image: victoriametrics/vmalert:v1.92.1
|
image: victoriametrics/vmalert:v1.93.0
|
||||||
depends_on:
|
depends_on:
|
||||||
- "victoriametrics"
|
- "victoriametrics"
|
||||||
- "alertmanager"
|
- "alertmanager"
|
||||||
|
|
|
@ -105,7 +105,7 @@ services:
|
||||||
- '--config=/config.yml'
|
- '--config=/config.yml'
|
||||||
|
|
||||||
vmsingle:
|
vmsingle:
|
||||||
image: victoriametrics/victoria-metrics:v1.92.1
|
image: victoriametrics/victoria-metrics:v1.93.0
|
||||||
ports:
|
ports:
|
||||||
- '8428:8428'
|
- '8428:8428'
|
||||||
command:
|
command:
|
||||||
|
|
|
@ -8,7 +8,7 @@
|
||||||
4. Set variables `DIGITALOCEAN_API_TOKEN` with `VM_VERSION` for `packer` environment and run make from example below:
|
4. Set variables `DIGITALOCEAN_API_TOKEN` with `VM_VERSION` for `packer` environment and run make from example below:
|
||||||
|
|
||||||
```console
|
```console
|
||||||
make release-victoria-metrics-digitalocean-oneclick-droplet DIGITALOCEAN_API_TOKEN="dop_v23_2e46f4759ceeeba0d0248" VM_VERSION="1.92.1"
|
make release-victoria-metrics-digitalocean-oneclick-droplet DIGITALOCEAN_API_TOKEN="dop_v23_2e46f4759ceeeba0d0248" VM_VERSION="1.93.0"
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -19,8 +19,8 @@ On the server:
|
||||||
* VictoriaMetrics is running on ports: 8428, 8089, 4242, 2003 and they are bound to the local interface.
|
* VictoriaMetrics is running on ports: 8428, 8089, 4242, 2003 and they are bound to the local interface.
|
||||||
|
|
||||||
********************************************************************************
|
********************************************************************************
|
||||||
# This image includes 1.92.1 version of VictoriaMetrics.
|
# This image includes 1.93.0 version of VictoriaMetrics.
|
||||||
# See Release notes https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.92.1
|
# See Release notes https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.93.0
|
||||||
|
|
||||||
# Welcome to VictoriaMetrics droplet!
|
# Welcome to VictoriaMetrics droplet!
|
||||||
|
|
||||||
|
|
|
@ -22,12 +22,25 @@ The following `tip` changes can be tested by building VictoriaMetrics components
|
||||||
* [How to build vmctl](https://docs.victoriametrics.com/vmctl.html#how-to-build)
|
* [How to build vmctl](https://docs.victoriametrics.com/vmctl.html#how-to-build)
|
||||||
|
|
||||||
|
|
||||||
## tip
|
## v1.93.x long-time support release (LTS)
|
||||||
|
|
||||||
|
## v1.93.1 long-time support release (LTS)
|
||||||
|
|
||||||
|
* BUGFIX: [storage](https://docs.victoriametrics.com/Single-server-VictoriaMetrics.html): properly set next retention time for indexDB. Previously it may enter into endless retention loop. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4873) for details.
|
||||||
|
* BUGFIX: do not allow starting VictoriaMetrics components with improperly set boolean command-line flags in the form `-boolFlagName value`, since this leads to silent incomplete flags' parsing. This form should be replaced with `-boolFlagName=value`. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4845).
|
||||||
|
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): properly set labels from `-remoteWrite.label` command-line flag just before sending samples to the configured `-remoteWrite.url` according to [these docs](https://docs.victoriametrics.com/vmagent.html#adding-labels-to-metrics). Previously these labels were incorrectly set before [the relabeling](https://docs.victoriametrics.com/vmagent.html#relabeling) configured via `-remoteWrite.urlRelabelConfigs` and [the stream aggregation](https://docs.victoriametrics.com/stream-aggregation.html) configured via `-remoteWrite.streamAggr.config`, so these labels could be lost or incorrectly transformed before sending the samples to remote storage. The fix allows using `-remoteWrite.label` for identifying `vmagent` instances in [cluster mode](https://docs.victoriametrics.com/vmagent.html#scraping-big-number-of-targets). See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4247) and [these docs](https://docs.victoriametrics.com/stream-aggregation.html#cluster-mode) for more details.
|
||||||
|
* BUGFIX: remove `DEBUG` logging when parsing `if` filters inside [relabeling rules](https://docs.victoriametrics.com/vmagent.html#relabeling-enhancements) and when parsing `match` filters inside [stream aggregation rules](https://docs.victoriametrics.com/stream-aggregation.html).
|
||||||
|
* BUGFIX: properly replace `:` chars in label names with `_` when `-usePromCompatibleNaming` command-line flag is passed to `vmagent`, `vminsert` or single-node VictoriaMetrics. This addresses [this comment](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/3113#issuecomment-1275077071).
|
||||||
|
* BUGFIX: [vmbackup](https://docs.victoriametrics.com/vmbackup.html): correctly check if specified `-dst` belongs to specified `-storageDataPath`. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4837).
|
||||||
|
* BUGFIX: [vmctl](https://docs.victoriametrics.com/vmctl.html): don't interrupt the migration process if no metrics were found for a specific tenant. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4796).
|
||||||
|
|
||||||
## [v1.93.0](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.93.0)
|
## [v1.93.0](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.93.0)
|
||||||
|
|
||||||
Released at 2023-08-12
|
Released at 2023-08-12
|
||||||
|
|
||||||
|
**v1.93.x is a line of LTS releases (e.g. long-time support). It contains important up-to-date bugfixes.
|
||||||
|
The v1.93.x line will be supported for at least 12 months since [v1.93.0](https://docs.victoriametrics.com/CHANGELOG.html#v1930) release**
|
||||||
|
|
||||||
**Update note**: starting from this release, [vmagent](https://docs.victoriametrics.com/vmagent.html) ignores timestamps provided by scrape targets by default - it associates scraped metrics with local timestamps instead. Set `honor_timestamps: true` in [scrape configs](https://docs.victoriametrics.com/sd_configs.html#scrape_configs) if timestamps provided by scrape targets must be used instead. This change helps removing gaps for metrics collected from [cadvisor](https://github.com/google/cadvisor) such as `container_memory_usage_bytes`. This also improves data compression and query performance over metrics collected from `cadvisor`. See more details [here](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4697).
|
**Update note**: starting from this release, [vmagent](https://docs.victoriametrics.com/vmagent.html) ignores timestamps provided by scrape targets by default - it associates scraped metrics with local timestamps instead. Set `honor_timestamps: true` in [scrape configs](https://docs.victoriametrics.com/sd_configs.html#scrape_configs) if timestamps provided by scrape targets must be used instead. This change helps removing gaps for metrics collected from [cadvisor](https://github.com/google/cadvisor) such as `container_memory_usage_bytes`. This also improves data compression and query performance over metrics collected from `cadvisor`. See more details [here](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4697).
|
||||||
|
|
||||||
* SECURITY: upgrade Go builder from Go1.20.6 to Go1.21.0 in order to fix [this issue](https://github.com/golang/go/issues/61460).
|
* SECURITY: upgrade Go builder from Go1.20.6 to Go1.21.0 in order to fix [this issue](https://github.com/golang/go/issues/61460).
|
||||||
|
@ -47,6 +60,7 @@ Released at 2023-08-12
|
||||||
* FEATURE: [Official Grafana dashboards for VictoriaMetrics](https://grafana.com/orgs/victoriametrics): correctly calculate `Bytes per point` value for single-server and cluster VM dashboards. Before, the calculation mistakenly accounted for the number of entries in indexdb in denominator, which could have shown lower values than expected.
|
* FEATURE: [Official Grafana dashboards for VictoriaMetrics](https://grafana.com/orgs/victoriametrics): correctly calculate `Bytes per point` value for single-server and cluster VM dashboards. Before, the calculation mistakenly accounted for the number of entries in indexdb in denominator, which could have shown lower values than expected.
|
||||||
* FEATURE: [Alerting rules for VictoriaMetrics](https://github.com/VictoriaMetrics/VictoriaMetrics/tree/master/deployment/docker#alerts): `ConcurrentFlushesHitTheLimit` alerting rule was moved from [single-server](https://github.com/VictoriaMetrics/VictoriaMetrics/blob/master/deployment/docker/alerts.yml) and [cluster](https://github.com/VictoriaMetrics/VictoriaMetrics/blob/master/deployment/docker/alerts-cluster.yml) alerts to the [list of "health" alerts](https://github.com/VictoriaMetrics/VictoriaMetrics/blob/master/deployment/docker/alerts-health.yml) as it could be related to many VictoriaMetrics components.
|
* FEATURE: [Alerting rules for VictoriaMetrics](https://github.com/VictoriaMetrics/VictoriaMetrics/tree/master/deployment/docker#alerts): `ConcurrentFlushesHitTheLimit` alerting rule was moved from [single-server](https://github.com/VictoriaMetrics/VictoriaMetrics/blob/master/deployment/docker/alerts.yml) and [cluster](https://github.com/VictoriaMetrics/VictoriaMetrics/blob/master/deployment/docker/alerts-cluster.yml) alerts to the [list of "health" alerts](https://github.com/VictoriaMetrics/VictoriaMetrics/blob/master/deployment/docker/alerts-health.yml) as it could be related to many VictoriaMetrics components.
|
||||||
|
|
||||||
|
* BUGFIX: [storage](https://docs.victoriametrics.com/Single-server-VictoriaMetrics.html): properly set next retention time for indexDB. Previously it may enter into endless retention loop. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4873) for details.
|
||||||
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): return human readable error if opentelemetry has json encoding. Follow-up after [PR](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/2570).
|
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): return human readable error if opentelemetry has json encoding. Follow-up after [PR](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/2570).
|
||||||
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): properly validate scheme for `proxy_url` field at the scrape config. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4811) for details.
|
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): properly validate scheme for `proxy_url` field at the scrape config. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4811) for details.
|
||||||
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): properly apply `if` filters during [relabeling](https://docs.victoriametrics.com/vmagent.html#relabeling-enhancements). Previously the `if` filter could improperly work. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4806) and [this pull request](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/4816).
|
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): properly apply `if` filters during [relabeling](https://docs.victoriametrics.com/vmagent.html#relabeling-enhancements). Previously the `if` filter could improperly work. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4806) and [this pull request](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/4816).
|
||||||
|
@ -386,6 +400,33 @@ Released at 2023-02-24
|
||||||
* BUGFIX: properly parse timestamps in milliseconds when [ingesting data via OpenTSDB telnet put protocol](https://docs.victoriametrics.com/#sending-data-via-telnet-put-protocol). Previously timestamps in milliseconds were mistakenly multiplied by 1000. Thanks to @Droxenator for the [pull request](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/3810).
|
* BUGFIX: properly parse timestamps in milliseconds when [ingesting data via OpenTSDB telnet put protocol](https://docs.victoriametrics.com/#sending-data-via-telnet-put-protocol). Previously timestamps in milliseconds were mistakenly multiplied by 1000. Thanks to @Droxenator for the [pull request](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/3810).
|
||||||
* BUGFIX: [MetricsQL](https://docs.victoriametrics.com/MetricsQL.html): do not add extrapolated points outside the real points when using [interpolate()](https://docs.victoriametrics.com/MetricsQL.html#interpolate) function. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/3816).
|
* BUGFIX: [MetricsQL](https://docs.victoriametrics.com/MetricsQL.html): do not add extrapolated points outside the real points when using [interpolate()](https://docs.victoriametrics.com/MetricsQL.html#interpolate) function. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/3816).
|
||||||
|
|
||||||
|
## [v1.87.7](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.87.7)
|
||||||
|
|
||||||
|
Released at 2023-08-12
|
||||||
|
|
||||||
|
**v1.87.x is a line of LTS releases (e.g. long-time support). It contains important up-to-date bugfixes.
|
||||||
|
The v1.87.x line will be supported for at least 12 months since [v1.87.0](https://docs.victoriametrics.com/CHANGELOG.html#v1870) release**
|
||||||
|
|
||||||
|
* SECURITY: upgrade Go builder from Go1.20.4 to Go1.21.0.
|
||||||
|
* SECURITY: upgrade base docker image (Alpine) from 3.18.2 to 3.18.3. See [alpine 3.18.3 release notes](https://alpinelinux.org/posts/Alpine-3.15.10-3.16.7-3.17.5-3.18.3-released.html).
|
||||||
|
|
||||||
|
* BUGFIX: vmselect: fix timestamp alignment for Prometheus querying API if time argument is less than 10m from the beginning of Unix epoch.
|
||||||
|
* BUGFIX: vminsert: fixed decoding of label values with slash when accepting data via [pushgateway protocol](https://docs.victoriametrics.com/#how-to-import-data-in-prometheus-exposition-format). This fixes Prometheus golang client compatibility. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4692).
|
||||||
|
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): properly validate scheme for `proxy_url` field at the scrape config. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4811) for details.
|
||||||
|
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): close HTTP connections to [service discovery](https://docs.victoriametrics.com/sd_configs.html) servers when they are no longer needed. This should prevent from possible connection exhasution in some cases. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4724).
|
||||||
|
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): properly apply `if` filters during [relabeling](https://docs.victoriametrics.com/vmagent.html#relabeling-enhancements). Previously the `if` filter could improperly work. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4806) and [this pull request](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/4816).
|
||||||
|
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): fix possible panic at shutdown when [stream aggregation](https://docs.victoriametrics.com/stream-aggregation.html) is enabled. See [this pull request](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/4407) for details.
|
||||||
|
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): use local scrape timestamps for the scraped metrics unless `honor_timestamps: true` option is explicitly set at [scrape_config](https://docs.victoriametrics.com/sd_configs.html#scrape_configs). This fixes gaps for metrics collected from [cadvisor](https://github.com/google/cadvisor) or similar exporters, which export metrics with invalid timestamps. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4697) and [this comment](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4697#issuecomment-1654614799) for details.
|
||||||
|
* BUGFIX: [vmauth](https://docs.victoriametrics.com/vmauth.html): Properly handle LOCAL command for proxy protocol. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/3335#issuecomment-1569864108).
|
||||||
|
* BUGFIX: [VictoriaMetrics cluster](https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html): properly return error from [/api/v1/query](https://docs.victoriametrics.com/keyConcepts.html#instant-query) and [/api/v1/query_range](https://docs.victoriametrics.com/keyConcepts.html#range-query) at `vmselect` when the `-search.maxSamplesPerQuery` or `-search.maxSamplesPerSeries` [limit](https://docs.victoriametrics.com/Cluster-VictoriaMetrics.html#resource-usage-limits) is exceeded. Previously incomplete response could be returned without the error if `vmselect` runs with `-replicationFactor` greater than 1. See [this pull request](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/4472).
|
||||||
|
* BUGFIX: [vmalert](https://docs.victoriametrics.com/vmalert.html): correctly calculate evaluation time for rules. Before, there was a low probability for discrepancy between actual time and rules evaluation time if evaluation interval was lower than the execution time for rules within the group.
|
||||||
|
* BUGFIX: [vmalert](https://docs.victoriametrics.com/vmalert.html): reset evaluation timestamp after modifying group interval. Before, there could have latency on rule evaluation time.
|
||||||
|
* BUGFIX: [vmalert](https://docs.victoriametrics.com/vmalert.html): Properly set datasource query params. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4340). Thanks to @gsakun for [the pull request](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/4341).
|
||||||
|
* BUGFIX: [vmalert](https://docs.victoriametrics.com/vmalert.html): Properly form path to static assets in WEB UI if `http.pathPrefix` set. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4349).
|
||||||
|
* BUGFIX: [vmalert](https://docs.victoriametrics.com/vmalert.html): properly return empty slices instead of nil for `/api/v1/rules` for groups with present name but absent `rules`. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4221).
|
||||||
|
* BUGFIX: [vmctl](https://docs.victoriametrics.com/vmctl.html): interrupt explore procedure in influx mode if vmctl found no numeric fields.
|
||||||
|
* BUGFIX: [vmctl](https://docs.victoriametrics.com/vmctl.html): fix panic in case `--remote-read-filter-time-start` flag is not set for remote-read mode. This flag is now required to use remote-read mode. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4553).
|
||||||
|
|
||||||
## [v1.87.6](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.87.6)
|
## [v1.87.6](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.87.6)
|
||||||
|
|
||||||
Released at 2023-05-18
|
Released at 2023-05-18
|
||||||
|
@ -660,6 +701,10 @@ See changes [here](https://docs.victoriametrics.com/CHANGELOG_2022.html#v1810)
|
||||||
|
|
||||||
See changes [here](https://docs.victoriametrics.com/CHANGELOG_2022.html#v1800)
|
See changes [here](https://docs.victoriametrics.com/CHANGELOG_2022.html#v1800)
|
||||||
|
|
||||||
|
## [v1.79.14](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.79.14)
|
||||||
|
|
||||||
|
See changes [here](https://docs.victoriametrics.com/CHANGELOG_2022.html#v17914)
|
||||||
|
|
||||||
## [v1.79.13](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.79.13)
|
## [v1.79.13](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.79.13)
|
||||||
|
|
||||||
See changes [here](https://docs.victoriametrics.com/CHANGELOG_2022.html#v17913)
|
See changes [here](https://docs.victoriametrics.com/CHANGELOG_2022.html#v17913)
|
||||||
|
|
|
@ -376,6 +376,22 @@ Released at 2022-08-08
|
||||||
* BUGFIX: [vmui](https://docs.victoriametrics.com/#vmui): properly show date picker at `Table` tab. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/2874).
|
* BUGFIX: [vmui](https://docs.victoriametrics.com/#vmui): properly show date picker at `Table` tab. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/2874).
|
||||||
* BUGFIX: properly generate http redirects if `-http.pathPrefix` command-line flag is set. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/2918).
|
* BUGFIX: properly generate http redirects if `-http.pathPrefix` command-line flag is set. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/2918).
|
||||||
|
|
||||||
|
## [v1.79.14](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.79.14)
|
||||||
|
|
||||||
|
Released at 2023-08-12
|
||||||
|
|
||||||
|
**v1.79.x is a line of LTS releases (e.g. long-time support). It contains important up-to-date bugfixes.
|
||||||
|
The v1.79.x line will be supported for at least 12 months since [v1.79.0](https://docs.victoriametrics.com/CHANGELOG.html#v1790) release**
|
||||||
|
|
||||||
|
* SECURITY: upgrade Go builder from Go1.20.4 to Go1.21.0.
|
||||||
|
* SECURITY: upgrade base docker image (Alpine) from 3.18.2 to 3.18.3. See [alpine 3.18.3 release notes](https://alpinelinux.org/posts/Alpine-3.15.10-3.16.7-3.17.5-3.18.3-released.html).
|
||||||
|
|
||||||
|
* BUGFIX: [vmagent](https://docs.victoriametrics.com/vmagent.html): properly apply `if` filters during [relabeling](https://docs.victoriametrics.com/vmagent.html#relabeling-enhancements). Previously the `if` filter could improperly work. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4806) and [this pull request](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/4816).
|
||||||
|
* BUGFIX: [vmalert](https://docs.victoriametrics.com/vmalert.html): Properly form path to static assets in WEB UI if `http.pathPrefix` set. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4349).
|
||||||
|
* BUGFIX: [vmalert](https://docs.victoriametrics.com/vmalert.html): Properly set datasource query params. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4340). Thanks to @gsakun for [the pull request](https://github.com/VictoriaMetrics/VictoriaMetrics/pull/4341).
|
||||||
|
* BUGFIX: [vmalert](https://docs.victoriametrics.com/vmalert.html): properly return empty slices instead of nil for `/api/v1/rules` and `/api/v1/alerts` API handlers. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4221).
|
||||||
|
* BUGFIX: [vmalert](https://docs.victoriametrics.com/vmalert.html): properly return empty slices instead of nil for `/api/v1/rules` for groups with present name but absent `rules`. See [this issue](https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4221).
|
||||||
|
|
||||||
## [v1.79.13](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.79.13)
|
## [v1.79.13](https://github.com/VictoriaMetrics/VictoriaMetrics/releases/tag/v1.79.13)
|
||||||
|
|
||||||
Released at 2023-05-18
|
Released at 2023-05-18
|
||||||
|
|
|
@ -176,7 +176,8 @@ VictoriaMetrics is developed at a fast pace, so it is recommended periodically c
|
||||||
|
|
||||||
### Environment variables
|
### Environment variables
|
||||||
|
|
||||||
All the VictoriaMetrics components allow referring environment variables in command-line flags via `%{ENV_VAR}` syntax.
|
All the VictoriaMetrics components allow referring environment variables in `yaml` configuration files (such as `-promscrape.config`)
|
||||||
|
and in command-line flags via `%{ENV_VAR}` syntax.
|
||||||
For example, `-metricsAuthKey=%{METRICS_AUTH_KEY}` is automatically expanded to `-metricsAuthKey=top-secret`
|
For example, `-metricsAuthKey=%{METRICS_AUTH_KEY}` is automatically expanded to `-metricsAuthKey=top-secret`
|
||||||
if `METRICS_AUTH_KEY=top-secret` environment variable exists at VictoriaMetrics startup.
|
if `METRICS_AUTH_KEY=top-secret` environment variable exists at VictoriaMetrics startup.
|
||||||
This expansion is performed by VictoriaMetrics itself.
|
This expansion is performed by VictoriaMetrics itself.
|
||||||
|
|
|
@ -184,7 +184,8 @@ VictoriaMetrics is developed at a fast pace, so it is recommended periodically c
|
||||||
|
|
||||||
### Environment variables
|
### Environment variables
|
||||||
|
|
||||||
All the VictoriaMetrics components allow referring environment variables in command-line flags via `%{ENV_VAR}` syntax.
|
All the VictoriaMetrics components allow referring environment variables in `yaml` configuration files (such as `-promscrape.config`)
|
||||||
|
and in command-line flags via `%{ENV_VAR}` syntax.
|
||||||
For example, `-metricsAuthKey=%{METRICS_AUTH_KEY}` is automatically expanded to `-metricsAuthKey=top-secret`
|
For example, `-metricsAuthKey=%{METRICS_AUTH_KEY}` is automatically expanded to `-metricsAuthKey=top-secret`
|
||||||
if `METRICS_AUTH_KEY=top-secret` environment variable exists at VictoriaMetrics startup.
|
if `METRICS_AUTH_KEY=top-secret` environment variable exists at VictoriaMetrics startup.
|
||||||
This expansion is performed by VictoriaMetrics itself.
|
This expansion is performed by VictoriaMetrics itself.
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
---
|
---
|
||||||
sort: 98
|
sort: 98
|
||||||
weight: 98
|
weight: 98
|
||||||
title: streaming aggregation
|
title: Streaming aggregation
|
||||||
menu:
|
menu:
|
||||||
docs:
|
docs:
|
||||||
parent: "victoriametrics"
|
parent: "victoriametrics"
|
||||||
|
@ -10,7 +10,7 @@ aliases:
|
||||||
- /stream-aggregation.html
|
- /stream-aggregation.html
|
||||||
---
|
---
|
||||||
|
|
||||||
# streaming aggregation
|
# Streaming aggregation
|
||||||
|
|
||||||
[vmagent](https://docs.victoriametrics.com/vmagent.html) and [single-node VictoriaMetrics](https://docs.victoriametrics.com/Single-server-VictoriaMetrics.html)
|
[vmagent](https://docs.victoriametrics.com/vmagent.html) and [single-node VictoriaMetrics](https://docs.victoriametrics.com/Single-server-VictoriaMetrics.html)
|
||||||
can aggregate incoming [samples](https://docs.victoriametrics.com/keyConcepts.html#raw-samples) in streaming mode by time and by labels before data is written to remote storage.
|
can aggregate incoming [samples](https://docs.victoriametrics.com/keyConcepts.html#raw-samples) in streaming mode by time and by labels before data is written to remote storage.
|
||||||
|
@ -673,3 +673,17 @@ support the following approaches for hot reloading stream aggregation configs fr
|
||||||
```
|
```
|
||||||
|
|
||||||
* By sending HTTP request to `/-/reload` endpoint (e.g. `http://vmagent:8429/-/reload` or `http://victoria-metrics:8428/-/reload).
|
* By sending HTTP request to `/-/reload` endpoint (e.g. `http://vmagent:8429/-/reload` or `http://victoria-metrics:8428/-/reload).
|
||||||
|
|
||||||
|
## Cluster mode
|
||||||
|
|
||||||
|
If you use [vmagent in cluster mode](https://docs.victoriametrics.com/vmagent.html#scraping-big-number-of-targets) for streaming aggregation
|
||||||
|
(with `-promscrape.cluster.*` parameters or with `VMAgent.spec.shardCount > 1` for [vmoperator](https://docs.victoriametrics.com/operator))
|
||||||
|
then be careful when aggregating metrics via `by`, `without` or modifying via `*_relabel_configs` parameters, since incorrect usage
|
||||||
|
may result in duplicates and data collision. For example, if more than one `vmagent` instance calculates `increase` for metric `http_requests_total`
|
||||||
|
with `by: [path]` directive, then all the `vmagent` instances will aggregate samples to the same set of time series with different `path` labels.
|
||||||
|
The proper fix would be to add an unique [`-remoteWrite.label`](https://docs.victoriametrics.com/vmagent.html#adding-labels-to-metrics) per each `vmagent`,
|
||||||
|
so every `vmagent` aggregates data into distinct set of time series. These time series then can be aggregated later as needed during querying.
|
||||||
|
|
||||||
|
For example, if `vmagent` instances run in Docker or Kubernetes, then you can refer `POD_NAME` or `HOSTNAME` environment variables
|
||||||
|
as an unique label value per each `vmagent`: `-remoteWrite.label='vmagent=%{HOSTNAME}` . See [these docs](https://docs.victoriametrics.com/#environment-variables)
|
||||||
|
on how to refer environment variables in VictoriaMetrics components.
|
||||||
|
|
|
@ -837,7 +837,7 @@ It is possible split migration process into set of smaller batches based on time
|
||||||
migrating large volumes of data as this adds indication of progress and ability to restore process from certain point
|
migrating large volumes of data as this adds indication of progress and ability to restore process from certain point
|
||||||
in case of failure.
|
in case of failure.
|
||||||
|
|
||||||
To use this you need to specify `--vm-native-step-interval` flag. Supported values are: `month`, `day`, `hour`, `minute`.
|
To use this you need to specify `--vm-native-step-interval` flag. Supported values are: `month`, `week`, `day`, `hour`, `minute`.
|
||||||
Note that in order to use this it is required `--vm-native-filter-time-start` to be set to calculate time ranges for
|
Note that in order to use this it is required `--vm-native-filter-time-start` to be set to calculate time ranges for
|
||||||
export process.
|
export process.
|
||||||
|
|
||||||
|
@ -847,7 +847,7 @@ Every range is being processed independently, which means that:
|
||||||
so it is possible to restart process starting from failed range.
|
so it is possible to restart process starting from failed range.
|
||||||
|
|
||||||
It is recommended using the `month` step when migrating the data over multiple months,
|
It is recommended using the `month` step when migrating the data over multiple months,
|
||||||
since the migration with `day` and `hour` steps may take longer time to complete because of additional overhead.
|
since the migration with `week`, `day` and `hour` steps may take longer time to complete because of additional overhead.
|
||||||
|
|
||||||
Usage example:
|
Usage example:
|
||||||
```console
|
```console
|
||||||
|
|
|
@ -32,6 +32,11 @@ func ParseFlagSet(fs *flag.FlagSet, args []string) {
|
||||||
// Do not use lib/logger here, since it is uninitialized yet.
|
// Do not use lib/logger here, since it is uninitialized yet.
|
||||||
log.Fatalf("cannot parse flags %q: %s", args, err)
|
log.Fatalf("cannot parse flags %q: %s", args, err)
|
||||||
}
|
}
|
||||||
|
if fs.NArg() > 0 {
|
||||||
|
// See https://github.com/VictoriaMetrics/VictoriaMetrics/issues/4845
|
||||||
|
log.Fatalf("unprocessed command-line args left: %s; the most likely reason is missing `=` between boolean flag name and value; "+
|
||||||
|
"see https://pkg.go.dev/flag#hdr-Command_line_flag_syntax", fs.Args())
|
||||||
|
}
|
||||||
if !*enable {
|
if !*enable {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
|
@ -80,7 +80,6 @@ func (ie *IfExpression) UnmarshalYAML(f func(interface{}) error) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (ie *IfExpression) unmarshalFromInterface(v interface{}) error {
|
func (ie *IfExpression) unmarshalFromInterface(v interface{}) error {
|
||||||
logger.Infof("DEBUG: unmarshaling ifExpr from %#v", v)
|
|
||||||
ies := ie.ies[:0]
|
ies := ie.ies[:0]
|
||||||
switch t := v.(type) {
|
switch t := v.(type) {
|
||||||
case string:
|
case string:
|
||||||
|
@ -89,7 +88,6 @@ func (ie *IfExpression) unmarshalFromInterface(v interface{}) error {
|
||||||
return fmt.Errorf("unexpected `match` option: %w", err)
|
return fmt.Errorf("unexpected `match` option: %w", err)
|
||||||
}
|
}
|
||||||
ies = append(ies, ieLocal)
|
ies = append(ies, ieLocal)
|
||||||
logger.Infof("DEBUG: unmarshaled ifExpr from %#v to %s", t, ieLocal)
|
|
||||||
case []interface{}:
|
case []interface{}:
|
||||||
for _, x := range t {
|
for _, x := range t {
|
||||||
s, ok := x.(string)
|
s, ok := x.(string)
|
||||||
|
@ -102,7 +100,6 @@ func (ie *IfExpression) unmarshalFromInterface(v interface{}) error {
|
||||||
}
|
}
|
||||||
ies = append(ies, ieLocal)
|
ies = append(ies, ieLocal)
|
||||||
}
|
}
|
||||||
logger.Infof("DEBUG: unmarshaled ifExpr from %#v to %s", t, ies)
|
|
||||||
default:
|
default:
|
||||||
return fmt.Errorf("unexpected `match` type; got %#v; want string or an array of strings", t)
|
return fmt.Errorf("unexpected `match` type; got %#v; want string or an array of strings", t)
|
||||||
}
|
}
|
||||||
|
|
|
@ -619,15 +619,28 @@ func fillLabelReferences(dst []byte, replacement string, labels []prompbmarshal.
|
||||||
return dst
|
return dst
|
||||||
}
|
}
|
||||||
|
|
||||||
// SanitizeName replaces unsupported by Prometheus chars in metric names and label names with _.
|
// SanitizeLabelName replaces unsupported by Prometheus chars in label names with _.
|
||||||
//
|
//
|
||||||
// See https://prometheus.io/docs/concepts/data_model/#metric-names-and-labels
|
// See https://prometheus.io/docs/concepts/data_model/#metric-names-and-labels
|
||||||
func SanitizeName(name string) string {
|
func SanitizeLabelName(name string) string {
|
||||||
return promSanitizer.Transform(name)
|
return labelNameSanitizer.Transform(name)
|
||||||
}
|
}
|
||||||
|
|
||||||
var promSanitizer = bytesutil.NewFastStringTransformer(func(s string) string {
|
var labelNameSanitizer = bytesutil.NewFastStringTransformer(func(s string) string {
|
||||||
return unsupportedPromChars.ReplaceAllString(s, "_")
|
return unsupportedLabelNameChars.ReplaceAllString(s, "_")
|
||||||
})
|
})
|
||||||
|
|
||||||
var unsupportedPromChars = regexp.MustCompile(`[^a-zA-Z0-9_:]`)
|
var unsupportedLabelNameChars = regexp.MustCompile(`[^a-zA-Z0-9_]`)
|
||||||
|
|
||||||
|
// SanitizeMetricName replaces unsupported by Prometheus chars in metric names with _.
|
||||||
|
//
|
||||||
|
// See https://prometheus.io/docs/concepts/data_model/#metric-names-and-labels
|
||||||
|
func SanitizeMetricName(value string) string {
|
||||||
|
return metricNameSanitizer.Transform(value)
|
||||||
|
}
|
||||||
|
|
||||||
|
var metricNameSanitizer = bytesutil.NewFastStringTransformer(func(s string) string {
|
||||||
|
return unsupportedMetricNameChars.ReplaceAllString(s, "_")
|
||||||
|
})
|
||||||
|
|
||||||
|
var unsupportedMetricNameChars = regexp.MustCompile(`[^a-zA-Z0-9_:]`)
|
||||||
|
|
|
@ -9,13 +9,13 @@ import (
|
||||||
"github.com/VictoriaMetrics/VictoriaMetrics/lib/promutils"
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/promutils"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestSanitizeName(t *testing.T) {
|
func TestSanitizeMetricName(t *testing.T) {
|
||||||
f := func(s, resultExpected string) {
|
f := func(s, resultExpected string) {
|
||||||
t.Helper()
|
t.Helper()
|
||||||
for i := 0; i < 5; i++ {
|
for i := 0; i < 5; i++ {
|
||||||
result := SanitizeName(s)
|
result := SanitizeMetricName(s)
|
||||||
if result != resultExpected {
|
if result != resultExpected {
|
||||||
t.Fatalf("unexpected result for SanitizeName(%q) at iteration %d; got %q; want %q", s, i, result, resultExpected)
|
t.Fatalf("unexpected result for SanitizeMetricName(%q) at iteration %d; got %q; want %q", s, i, result, resultExpected)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -25,6 +25,22 @@ func TestSanitizeName(t *testing.T) {
|
||||||
f("foo...bar", "foo___bar")
|
f("foo...bar", "foo___bar")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestSanitizeLabelName(t *testing.T) {
|
||||||
|
f := func(s, resultExpected string) {
|
||||||
|
t.Helper()
|
||||||
|
for i := 0; i < 5; i++ {
|
||||||
|
result := SanitizeLabelName(s)
|
||||||
|
if result != resultExpected {
|
||||||
|
t.Fatalf("unexpected result for SanitizeLabelName(%q) at iteration %d; got %q; want %q", s, i, result, resultExpected)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
f("", "")
|
||||||
|
f("a", "a")
|
||||||
|
f("foo.bar/baz:a", "foo_bar_baz_a")
|
||||||
|
f("foo...bar", "foo___bar")
|
||||||
|
}
|
||||||
|
|
||||||
func TestLabelsToString(t *testing.T) {
|
func TestLabelsToString(t *testing.T) {
|
||||||
f := func(labels []prompbmarshal.Label, sExpected string) {
|
f := func(labels []prompbmarshal.Label, sExpected string) {
|
||||||
t.Helper()
|
t.Helper()
|
||||||
|
|
|
@ -8,20 +8,39 @@ import (
|
||||||
"github.com/VictoriaMetrics/VictoriaMetrics/lib/prompbmarshal"
|
"github.com/VictoriaMetrics/VictoriaMetrics/lib/prompbmarshal"
|
||||||
)
|
)
|
||||||
|
|
||||||
func BenchmarkSanitizeName(b *testing.B) {
|
func BenchmarkSanitizeMetricName(b *testing.B) {
|
||||||
for _, name := range []string{"", "foo", "foo-bar-baz", "http_requests_total"} {
|
for _, name := range []string{"", "foo", "foo-bar-baz", "http_requests_total"} {
|
||||||
b.Run(name, func(b *testing.B) {
|
b.Run(name, func(b *testing.B) {
|
||||||
benchmarkSanitizeName(b, name)
|
benchmarkSanitizeMetricName(b, name)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func benchmarkSanitizeName(b *testing.B, name string) {
|
func benchmarkSanitizeMetricName(b *testing.B, name string) {
|
||||||
b.ReportAllocs()
|
b.ReportAllocs()
|
||||||
b.SetBytes(1)
|
b.SetBytes(1)
|
||||||
b.RunParallel(func(pb *testing.PB) {
|
b.RunParallel(func(pb *testing.PB) {
|
||||||
for pb.Next() {
|
for pb.Next() {
|
||||||
sanitizedName := SanitizeName(name)
|
sanitizedName := SanitizeMetricName(name)
|
||||||
|
GlobalSink += len(sanitizedName)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func BenchmarkSanitizeLabelName(b *testing.B) {
|
||||||
|
for _, name := range []string{"", "foo", "foo-bar-baz", "http_requests_total"} {
|
||||||
|
b.Run(name, func(b *testing.B) {
|
||||||
|
benchmarkSanitizeLabelName(b, name)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func benchmarkSanitizeLabelName(b *testing.B, name string) {
|
||||||
|
b.ReportAllocs()
|
||||||
|
b.SetBytes(1)
|
||||||
|
b.RunParallel(func(pb *testing.PB) {
|
||||||
|
for pb.Next() {
|
||||||
|
sanitizedName := SanitizeLabelName(name)
|
||||||
GlobalSink += len(sanitizedName)
|
GlobalSink += len(sanitizedName)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
|
@ -757,7 +757,7 @@ func (s *Storage) mustRotateIndexDB(currentTime time.Time) {
|
||||||
idbNew := mustOpenIndexDB(idbNewPath, s, &s.isReadOnly)
|
idbNew := mustOpenIndexDB(idbNewPath, s, &s.isReadOnly)
|
||||||
|
|
||||||
// Update nextRotationTimestamp
|
// Update nextRotationTimestamp
|
||||||
nextRotationTimestamp := currentTime.UnixMilli() + s.retentionMsecs/1000
|
nextRotationTimestamp := currentTime.Unix() + s.retentionMsecs/1000
|
||||||
atomic.StoreInt64(&s.nextRotationTimestamp, nextRotationTimestamp)
|
atomic.StoreInt64(&s.nextRotationTimestamp, nextRotationTimestamp)
|
||||||
|
|
||||||
// Set idbNext to idbNew
|
// Set idbNext to idbNew
|
||||||
|
|
Loading…
Reference in a new issue