Skip to content

Commit

Permalink
internal/envflag: support deprecated flags
Browse files Browse the repository at this point in the history
We wish to remove support for disabling the modules experiment
but we do not wish to break users who have set `CUE_EXPERIMENT=modules=1`
to use the experiment.

So change `envflag` to support marking a field as deprecated,
meaning that any attempt to change it from its default value
will result in an error.

Also in passing remove some unnecessary type parameters.

Signed-off-by: Roger Peppe <rogpeppe@gmail.com>
Change-Id: Ifd552d9212cda1516343d338a6dcc30f1febe150
Dispatch-Trailer: {"type":"trybot","CL":1200577,"patchset":1,"ref":"refs/changes/77/1200577/1","targetBranch":"master"}
  • Loading branch information
rogpeppe authored and cueckoo committed Sep 3, 2024
1 parent e2685dd commit d1c20bf
Show file tree
Hide file tree
Showing 2 changed files with 63 additions and 13 deletions.
43 changes: 32 additions & 11 deletions internal/envflag/flag.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,10 @@ func Init[T any](flags *T, envVar string) error {
// The struct field tag may contain a default value other than the zero value,
// such as `envflag:"default:true"` to set a boolean field to true by default.
//
// The tag may be marked as deprecated with `envflag:"deprecated"`
// which will cause Parse to return an error if the user attempts to set
// its value to anything but the default value.
//
// The string may contain a comma-separated list of name=value pairs values
// representing the boolean fields in the struct type T. If the value is omitted
// entirely, the value is assumed to be name=true.
Expand All @@ -34,25 +38,32 @@ func Init[T any](flags *T, envVar string) error {
func Parse[T any](flags *T, env string) error {
// Collect the field indices and set the default values.
indexByName := make(map[string]int)
deprecated := make(map[string]bool)
fv := reflect.ValueOf(flags).Elem()
ft := fv.Type()
for i := 0; i < ft.NumField(); i++ {
field := ft.Field(i)
name := strings.ToLower(field.Name)
defaultValue := false
if tagStr, ok := field.Tag.Lookup("envflag"); ok {
defaultStr, ok := strings.CutPrefix(tagStr, "default:")
// TODO: consider panicking for these error types.
if !ok {
return fmt.Errorf("expected tag like `envflag:\"default:true\"`: %s", tagStr)
}
v, err := strconv.ParseBool(defaultStr)
if err != nil {
return fmt.Errorf("invalid default bool value for %s: %v", field.Name, err)
for _, f := range strings.Split(tagStr, ",") {
key, rest, _ := strings.Cut(f, ":")
switch key {
case "default":
v, err := strconv.ParseBool(rest)
if err != nil {
return fmt.Errorf("invalid default bool value for %s: %v", field.Name, err)
}
defaultValue = v
fv.Field(i).SetBool(defaultValue)
case "deprecated":
deprecated[name] = true
default:
return fmt.Errorf("unknown envflag tag %q", f)
}
}
defaultValue = v
fv.Field(i).SetBool(defaultValue)
}
indexByName[strings.ToLower(field.Name)] = i
indexByName[name] = i
}

if env == "" {
Expand Down Expand Up @@ -80,6 +91,16 @@ func Parse[T any](flags *T, env string) error {
errs = append(errs, fmt.Errorf("unknown %s", elem))
continue
}
if deprecated[name] {
// We allow setting deprecated flags to their default value so
// that bold explorers will not be penalised for their
// experimentation.
if fv.Field(index).Bool() != value {
errs = append(errs, fmt.Errorf("cannot set deprecated flag %q", name))
}
continue
}

fv.Field(index).SetBool(value)
}
return errors.Join(errs...)
Expand Down
33 changes: 31 additions & 2 deletions internal/envflag/flag_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,11 @@ type testFlags struct {
DefaultTrue bool `envflag:"default:true"`
}

type deprecatedFlags struct {
Foo bool `envflag:"deprecated"`
Bar bool `envflag:"deprecated,default:true"`
}

func success[T comparable](want T) func(t *testing.T) {
return func(t *testing.T) {
var x T
Expand Down Expand Up @@ -54,7 +59,7 @@ var tests = []struct {
}, {
testName: "Unknown",
envVal: "ratchet",
test: failure[testFlags](testFlags{DefaultTrue: true},
test: failure(testFlags{DefaultTrue: true},
"cannot parse TEST_VAR: unknown ratchet"),
}, {
testName: "Set",
Expand All @@ -73,7 +78,7 @@ var tests = []struct {
}, {
testName: "SetWithUnknown",
envVal: "foo,other",
test: failure[testFlags](testFlags{
test: failure(testFlags{
Foo: true,
DefaultTrue: true,
}, "cannot parse TEST_VAR: unknown other"),
Expand Down Expand Up @@ -108,6 +113,30 @@ var tests = []struct {
testName: "Invalid",
envVal: "foo=2,BarBaz=true",
test: invalid(testFlags{DefaultTrue: true}),
}, {
testName: "DeprecatedWithFalseDefault",
envVal: "foo=1",
test: failure(deprecatedFlags{
Bar: true,
}, `cannot parse TEST_VAR: cannot set deprecated flag "foo"`),
}, {
testName: "DeprecatedNoopWhenSameAndFalseDefault",
envVal: "foo=false",
test: success(deprecatedFlags{
Bar: true,
}),
}, {
testName: "DeprecatedWithTrueDefault",
envVal: "bar=0",
test: failure(deprecatedFlags{
Bar: true,
}, `cannot parse TEST_VAR: cannot set deprecated flag "bar"`),
}, {
testName: "DeprecatedNoopWhenSameAndTrueDefault",
envVal: "bar=1",
test: success(deprecatedFlags{
Bar: true,
}),
}}

func TestInit(t *testing.T) {
Expand Down

0 comments on commit d1c20bf

Please sign in to comment.