Skip to content

Commit

Permalink
config: Allow matching errors returned from Validate()
Browse files Browse the repository at this point in the history
Any error returned from `Validate()` is propagated with
`ErrInvalidConfiguration` attached, allowing `errors.Is()` checks on the
returned errors to recognize both `ErrInvalidConfiguration` and the original
errors returned from `Validate()`.
  • Loading branch information
lippserd committed Oct 25, 2024
1 parent 80e4657 commit 7e61437
Showing 1 changed file with 20 additions and 2 deletions.
22 changes: 20 additions & 2 deletions config/config.go
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@ import (
"github.com/caarlos0/env/v11"
"github.com/creasty/defaults"
"github.com/goccy/go-yaml"
"github.com/icinga/icinga-go-library/multierr"
"github.com/jessevdk/go-flags"
"github.com/pkg/errors"
"os"
Expand All @@ -60,12 +61,23 @@ import (
// must be a non-nil struct pointer.
var ErrInvalidArgument = stderrors.New("invalid argument")

// ErrInvalidConfiguration is attached to errors returned by [FromYAMLFile] or [FromEnv] when
// the configuration is invalid,
// i.e. if the Validate method of the provided [Validator] interface returns an error,
// which is then propagated by these functions.
// Note that for such errors, errors.Is() will recognize both ErrInvalidConfiguration and
// the original errors returned from Validate.
var ErrInvalidConfiguration = stderrors.New("invalid configuration")

// FromYAMLFile parses the given YAML file and stores the result
// in the value pointed to by v. If v is nil or not a struct pointer,
// FromYAMLFile returns an [ErrInvalidArgument] error.
// It is possible to define default values via the struct tag `default`.
// The function also validates the configuration using the Validate method
// of the provided [Validator] interface.
// Any error returned from Validate is propagated with ErrInvalidConfiguration attached,
// allowing errors.Is() checks on the returned errors to recognize both ErrInvalidConfiguration and
// the original errors returned from Validate.
//
// Example usage:
//
Expand Down Expand Up @@ -114,7 +126,7 @@ func FromYAMLFile(name string, v Validator) error {
}

if err := v.Validate(); err != nil {
return errors.Wrap(err, "invalid configuration")
return multierr.Wrap(err, ErrInvalidConfiguration)
}

return nil
Expand All @@ -125,6 +137,12 @@ type EnvOptions = env.Options

// FromEnv parses environment variables and stores the result in the value pointed to by v.
// If v is nil or not a struct pointer, FromEnv returns an [ErrInvalidArgument] error.
// It is possible to define default values via the struct tag `default`.
// The function also validates the configuration using the Validate method
// of the provided [Validator] interface.
// Any error returned from Validate is propagated with ErrInvalidConfiguration attached,
// allowing errors.Is() checks on the returned errors to recognize both ErrInvalidConfiguration and
// the original errors returned from Validate.
func FromEnv(v Validator, options EnvOptions) error {
if err := validateNonNilStructPointer(v); err != nil {
return errors.WithStack(err)
Expand All @@ -139,7 +157,7 @@ func FromEnv(v Validator, options EnvOptions) error {
}

if err := v.Validate(); err != nil {
return errors.Wrap(err, "invalid configuration")
return multierr.Wrap(err, ErrInvalidConfiguration)
}

return nil
Expand Down

0 comments on commit 7e61437

Please sign in to comment.