Files
argo-cd/util/env/env.go
Matthieu MOREL 34eeede822 chore(util): Fix modernize linter (#26329)
Signed-off-by: Matthieu MOREL <matthieu.morel35@gmail.com>
2026-02-09 00:44:05 -10:00

221 lines
6.8 KiB
Go

package env
import (
"math"
"os"
"strconv"
"strings"
"time"
timeutil "github.com/argoproj/pkg/time"
log "github.com/sirupsen/logrus"
)
// Helper function to parse a number from an environment variable. Returns a
// default if env is not set, is not parseable to a number, exceeds maximum (if
// maximum is greater than 0) or is less than minimum.
func ParseNumFromEnv(env string, defaultValue, minimum, maximum int) int {
str := os.Getenv(env)
if str == "" {
return defaultValue
}
num, err := strconv.ParseInt(str, 10, 0)
if err != nil {
log.Warnf("Could not parse '%s' as a number from environment %s", str, env)
return defaultValue
}
if num > math.MaxInt || num < math.MinInt {
log.Warnf("Value in %s is %d is outside of the min and max %d allowed values. Using default %d", env, num, minimum, defaultValue)
return defaultValue
}
if int(num) < minimum {
log.Warnf("Value in %s is %d, which is less than minimum %d allowed", env, num, minimum)
return defaultValue
}
if int(num) > maximum {
log.Warnf("Value in %s is %d, which is greater than maximum %d allowed", env, num, maximum)
return defaultValue
}
return int(num)
}
// Helper function to parse a int64 from an environment variable. Returns a
// default if env is not set, is not parseable to a number, exceeds maximum (if
// maximum is greater than 0) or is less than minimum.
func ParseInt64FromEnv(env string, defaultValue, minimum, maximum int64) int64 {
str := os.Getenv(env)
if str == "" {
return defaultValue
}
num, err := strconv.ParseInt(str, 10, 64)
if err != nil {
log.Warnf("Could not parse '%s' as a int64 from environment %s", str, env)
return defaultValue
}
if num < minimum {
log.Warnf("Value in %s is %d, which is less than minimum %d allowed", env, num, minimum)
return defaultValue
}
if num > maximum {
log.Warnf("Value in %s is %d, which is greater than maximum %d allowed", env, num, maximum)
return defaultValue
}
return num
}
// Helper function to parse a float32 from an environment variable. Returns a
// default if env is not set, is not parseable to a number, exceeds maximum (if
// maximum is greater than 0) or is less than minimum (and minimum is greater than 0).
func ParseFloatFromEnv(env string, defaultValue, minimum, maximum float32) float32 {
str := os.Getenv(env)
if str == "" {
return defaultValue
}
num, err := strconv.ParseFloat(str, 32)
if err != nil {
log.Warnf("Could not parse '%s' as a float32 from environment %s", str, env)
return defaultValue
}
if float32(num) < minimum {
log.Warnf("Value in %s is %f, which is less than minimum %f allowed", env, num, minimum)
return defaultValue
}
if float32(num) > maximum {
log.Warnf("Value in %s is %f, which is greater than maximum %f allowed", env, num, maximum)
return defaultValue
}
return float32(num)
}
// Helper function to parse a float64 from an environment variable. Returns a
// default if env is not set, is not parseable to a number, exceeds maximum (if
// maximum is greater than 0) or is less than minimum (and minimum is greater than 0).
func ParseFloat64FromEnv(env string, defaultValue, minimum, maximum float64) float64 {
str := os.Getenv(env)
if str == "" {
return defaultValue
}
num, err := strconv.ParseFloat(str, 64)
if err != nil {
log.Warnf("Could not parse '%s' as a float64 from environment %s", str, env)
return defaultValue
}
if num < minimum {
log.Warnf("Value in %s is %f, which is less than minimum %f allowed", env, num, minimum)
return defaultValue
}
if num > maximum {
log.Warnf("Value in %s is %f, which is greater than maximum %f allowed", env, num, maximum)
return defaultValue
}
return num
}
// Helper function to parse a time duration from an environment variable. Returns a
// default if env is not set, is not parseable to a duration, exceeds maximum (if
// maximum is greater than 0) or is less than minimum.
//
// nolint:unparam
func ParseDurationFromEnv(env string, defaultValue, minimum, maximum time.Duration) time.Duration {
str := os.Getenv(env)
if str == "" {
return defaultValue
}
dur, err := time.ParseDuration(str)
if err != nil {
// provides backwards compatibility for durations defined in days, see: https://github.com/argoproj/argo-cd/issues/24740
durPtr, err2 := timeutil.ParseDuration(str)
if err2 != nil {
log.Warnf("Could not parse '%s' as a duration from environment %s", str, env)
return defaultValue
}
dur = *durPtr
}
if dur < minimum {
log.Warnf("Value in %s is %s, which is less than minimum %s allowed", env, dur, minimum)
return defaultValue
}
if dur > maximum {
log.Warnf("Value in %s is %s, which is greater than maximum %s allowed", env, dur, maximum)
return defaultValue
}
return dur
}
type StringFromEnvOpts struct {
// AllowEmpty allows the value to be empty as long as the environment variable is set.
AllowEmpty bool
}
func StringFromEnv(env string, defaultValue string, opts ...StringFromEnvOpts) string {
opt := StringFromEnvOpts{}
for _, o := range opts {
opt.AllowEmpty = opt.AllowEmpty || o.AllowEmpty
}
if str, ok := os.LookupEnv(env); opt.AllowEmpty && ok || str != "" {
return str
}
return defaultValue
}
// StringsFromEnv parses given value from the environment as a list of strings,
// using separator as the delimeter, and returns them as a slice. The strings
// in the returned slice will have leading and trailing white space removed.
func StringsFromEnv(env string, defaultValue []string, separator string) []string {
if str := os.Getenv(env); str != "" {
ss := strings.Split(str, separator)
for i, s := range ss {
ss[i] = strings.TrimSpace(s)
}
return ss
}
return defaultValue
}
// ParseBoolFromEnv retrieves a boolean value from given environment envVar.
// Returns default value if envVar is not set.
//
// nolint:unparam
func ParseBoolFromEnv(envVar string, defaultValue bool) bool {
if val := os.Getenv(envVar); val != "" {
if strings.EqualFold(val, "true") {
return true
} else if strings.EqualFold(val, "false") {
return false
}
}
return defaultValue
}
// ParseStringToStringFromEnv parses given value from the environment as a map of string.
// Returns default value if envVar is not set.
func ParseStringToStringFromEnv(envVar string, defaultValue map[string]string, separator string) map[string]string {
str := os.Getenv(envVar)
str = strings.TrimSpace(str)
if str == "" {
return defaultValue
}
parsed := make(map[string]string)
for pair := range strings.SplitSeq(str, separator) {
keyvalue := strings.Split(pair, "=")
if len(keyvalue) != 2 {
log.Warnf("Invalid key-value pair when parsing environment '%s' as a string map", str)
return defaultValue
}
key := strings.TrimSpace(keyvalue[0])
value := strings.TrimSpace(keyvalue[1])
if _, ok := parsed[key]; ok {
log.Warnf("Duplicate key '%s' when parsing environment '%s' as a string map", key, str)
return defaultValue
}
parsed[key] = value
}
return parsed
}