mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2026-05-17 08:36:55 +03:00
224 lines
6.5 KiB
Go
224 lines
6.5 KiB
Go
package flagutil
|
|
|
|
import (
|
|
"encoding/json"
|
|
"flag"
|
|
"fmt"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/VictoriaMetrics/metricsql"
|
|
)
|
|
|
|
// NewRetentionDuration returns new `duration` flag with the given name, defaultValue and description.
|
|
//
|
|
// DefaultValue is in months.
|
|
func NewRetentionDuration(name string, defaultValue string, description string) *RetentionDuration {
|
|
description += "\nThe following optional suffixes are supported: s (second), h (hour), d (day), w (week), M (month), y (year). " +
|
|
"If suffix isn't set, then the duration is counted in months"
|
|
d := &RetentionDuration{}
|
|
if err := d.Set(defaultValue); err != nil {
|
|
panic(fmt.Sprintf("BUG: can not parse default value %s for flag %s", defaultValue, name))
|
|
}
|
|
flag.Var(d, name, description)
|
|
return d
|
|
}
|
|
|
|
// RetentionDuration is a flag for holding duration for retention period.
|
|
type RetentionDuration struct {
|
|
// msecs contains parsed duration in milliseconds.
|
|
msecs int64
|
|
|
|
valueString string
|
|
}
|
|
|
|
var (
|
|
_ json.Marshaler = (*RetentionDuration)(nil)
|
|
_ json.Unmarshaler = (*RetentionDuration)(nil)
|
|
)
|
|
|
|
// MarshalJSON implements json.Marshaler interface
|
|
func (d *RetentionDuration) MarshalJSON() ([]byte, error) {
|
|
return json.Marshal(d.valueString)
|
|
}
|
|
|
|
// UnmarshalJSON implements json.Unmarshaler interface
|
|
func (d *RetentionDuration) UnmarshalJSON(data []byte) error {
|
|
var s string
|
|
if err := json.Unmarshal(data, &s); err != nil {
|
|
return err
|
|
}
|
|
return d.Set(s)
|
|
}
|
|
|
|
// Duration returns d as time.Duration
|
|
func (d *RetentionDuration) Duration() time.Duration {
|
|
return time.Millisecond * time.Duration(d.msecs)
|
|
}
|
|
|
|
// Milliseconds returns d in milliseconds
|
|
func (d *RetentionDuration) Milliseconds() int64 {
|
|
return d.msecs
|
|
}
|
|
|
|
// String implements flag.Value interface
|
|
func (d *RetentionDuration) String() string {
|
|
return d.valueString
|
|
}
|
|
|
|
// Set implements flag.Value interface
|
|
// It assumes that value without unit should be parsed as `month` duration.
|
|
// It returns an error if value has `m` unit.
|
|
func (d *RetentionDuration) Set(value string) error {
|
|
if value == "" {
|
|
d.msecs = 0
|
|
d.valueString = ""
|
|
return nil
|
|
}
|
|
|
|
// An attempt to parse value as months with unit M(onth).
|
|
if cutValue, found := strings.CutSuffix(value, "M"); found {
|
|
months, err := strconv.ParseFloat(cutValue, 64)
|
|
if err != nil {
|
|
return fmt.Errorf("cannot parse months from %q: %w", value, err)
|
|
}
|
|
|
|
if months > maxMonths {
|
|
return fmt.Errorf("duration months must be smaller than %d; got %g", maxMonths, months)
|
|
}
|
|
if months < 0 {
|
|
return fmt.Errorf("duration months cannot be negative; got %g", months)
|
|
}
|
|
d.msecs = int64(months * msecsPer31Days)
|
|
d.valueString = value
|
|
return nil
|
|
}
|
|
|
|
// An attempt to parse value as a numeric month (without unit).
|
|
// Such values should be treated as months for BC and historical reasons.
|
|
// The format is deprecated, value with unit M should be used instead.
|
|
months, err := strconv.ParseFloat(value, 64)
|
|
if err == nil {
|
|
if months > maxMonths {
|
|
return fmt.Errorf("duration months must be smaller than %d; got %g", maxMonths, months)
|
|
}
|
|
if months < 0 {
|
|
return fmt.Errorf("duration months cannot be negative; got %g", months)
|
|
}
|
|
d.msecs = int64(months * msecsPer31Days)
|
|
d.valueString = value
|
|
return nil
|
|
}
|
|
|
|
// Parse duration.
|
|
value = strings.ToLower(value)
|
|
if strings.HasSuffix(value, "m") {
|
|
return fmt.Errorf("duration in months must be set with capital `M` suffix, lower case `m` means minutes and not allowed; got %s", value)
|
|
}
|
|
msecs, err := metricsql.PositiveDurationValue(value, 0)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if msecs/msecsPer31Days > maxMonths {
|
|
return fmt.Errorf("duration must be smaller than %d months; got approx %d months", maxMonths, msecs/msecsPer31Days)
|
|
}
|
|
d.msecs = msecs
|
|
d.valueString = value
|
|
return nil
|
|
}
|
|
|
|
// NewExtendedDuration returns new `duration` flag with the given name, defaultValue and description.
|
|
//
|
|
// Requires explicit unit suffixes (s, m, h, d, w, y). Bare numbers without units will cause errors (except 0).
|
|
func NewExtendedDuration(name string, defaultValue string, description string) *ExtendedDuration {
|
|
description += "\nThe following unit suffixes are required: s (second), m (minute), h (hour), d (day), w (week), y (year). " +
|
|
"Bare numbers without units are not allowed (except 0)"
|
|
d := &ExtendedDuration{}
|
|
if err := d.Set(defaultValue); err != nil {
|
|
panic(fmt.Sprintf("BUG: can not parse default value %s for flag %s", defaultValue, name))
|
|
}
|
|
flag.Var(d, name, description)
|
|
return d
|
|
}
|
|
|
|
// ExtendedDuration is a flag for specifying time durations with explicit unit suffixes.
|
|
// Unlike RetentionDuration, it requires explicit unit suffixes and doesn't treat bare numbers as months.
|
|
// Supported units: s (seconds), m (minutes), h (hours), d (days), w (weeks), y (years).
|
|
type ExtendedDuration struct {
|
|
// msecs contains parsed duration in milliseconds.
|
|
msecs int64
|
|
|
|
valueString string
|
|
}
|
|
|
|
var (
|
|
_ json.Marshaler = (*ExtendedDuration)(nil)
|
|
_ json.Unmarshaler = (*ExtendedDuration)(nil)
|
|
)
|
|
|
|
// MarshalJSON implements json.Marshaler interface
|
|
func (d *ExtendedDuration) MarshalJSON() ([]byte, error) {
|
|
return json.Marshal(d.valueString)
|
|
}
|
|
|
|
// UnmarshalJSON implements json.Unmarshaler interface
|
|
func (d *ExtendedDuration) UnmarshalJSON(data []byte) error {
|
|
var s string
|
|
if err := json.Unmarshal(data, &s); err != nil {
|
|
return err
|
|
}
|
|
return d.Set(s)
|
|
}
|
|
|
|
// Duration returns d as time.Duration
|
|
func (d *ExtendedDuration) Duration() time.Duration {
|
|
return time.Millisecond * time.Duration(d.msecs)
|
|
}
|
|
|
|
// Milliseconds returns d in milliseconds
|
|
func (d *ExtendedDuration) Milliseconds() int64 {
|
|
return d.msecs
|
|
}
|
|
|
|
// String implements flag.Value interface
|
|
func (d *ExtendedDuration) String() string {
|
|
return d.valueString
|
|
}
|
|
|
|
// Set implements flag.Value interface
|
|
// It requires explicit unit suffixes and rejects bare numbers (except 0).
|
|
func (d *ExtendedDuration) Set(value string) error {
|
|
if value == "" {
|
|
d.msecs = 0
|
|
d.valueString = ""
|
|
return nil
|
|
}
|
|
|
|
// Check for bare numbers first
|
|
if f, err := strconv.ParseFloat(value, 64); err == nil {
|
|
// It's a bare number
|
|
if f != 0 {
|
|
return fmt.Errorf("duration value must have a unit suffix (s, m, h, d, w, y); got bare number %q (0 is allowed)", value)
|
|
}
|
|
// Allow 0 as it's unambiguous
|
|
d.msecs = 0
|
|
d.valueString = value
|
|
return nil
|
|
}
|
|
|
|
// Parse duration with units using metricsql
|
|
value = strings.ToLower(value)
|
|
msecs, err := metricsql.PositiveDurationValue(value, 0)
|
|
if err != nil {
|
|
return fmt.Errorf("cannot parse duration %q: %w", value, err)
|
|
}
|
|
|
|
d.msecs = msecs
|
|
d.valueString = value
|
|
return nil
|
|
}
|
|
|
|
const maxMonths = 12 * 100
|
|
const msecsPer31Days = 31 * 24 * 3600 * 1000
|