Documentation ¶
Overview ¶
Package ucfg provides a common representation for hierarchical configurations.
The common representation provided by the Config type can be used with different configuration file formats like XML, JSON, HSJSON, YAML, or TOML.
Config provides a low level and a high level interface for reading settings with additional features like custom unpackers, validation and capturing sub-configurations for deferred interpretation, lazy intra-configuration variable expansion, and OS environment variable expansion.
Index ¶
- Constants
- Variables
- func DecodeSecretString(s string) (string, string)
- func RegisterValidator(name string, cb ValidatorCallback) error
- type BoolUnpacker
- type Config
- func (c *Config) Bool(name string, idx int, opts ...Option) (bool, error)
- func (c *Config) Child(name string, idx int, opts ...Option) (*Config, error)
- func (c *Config) CountField(name string, opts ...Option) (int, error)
- func (c *Config) FlattenedKeys(opts ...Option) []string
- func (c *Config) Float(name string, idx int, opts ...Option) (float64, error)
- func (c *Config) GetFields() []string
- func (c *Config) Has(name string, idx int, options ...Option) (bool, error)
- func (c *Config) HasField(name string) bool
- func (c *Config) Int(name string, idx int, opts ...Option) (int64, error)
- func (c *Config) IsArray() bool
- func (c *Config) IsDict() bool
- func (c *Config) Merge(from interface{}, options ...Option) error
- func (c *Config) Parent() *Config
- func (c *Config) Path(sep string) string
- func (c *Config) PathOf(field, sep string) string
- func (c *Config) Remove(name string, idx int, options ...Option) (bool, error)
- func (c *Config) SetBool(name string, idx int, value bool, opts ...Option) error
- func (c *Config) SetChild(name string, idx int, value *Config, opts ...Option) error
- func (c *Config) SetFloat(name string, idx int, value float64, opts ...Option) error
- func (c *Config) SetInt(name string, idx int, value int64, opts ...Option) error
- func (c *Config) SetString(name string, idx int, value string, opts ...Option) error
- func (c *Config) SetUint(name string, idx int, value uint64, opts ...Option) error
- func (c *Config) String(name string, idx int, opts ...Option) (string, error)
- func (c *Config) Uint(name string, idx int, opts ...Option) (uint64, error)
- func (c *Config) Unpack(to interface{}, options ...Option) error
- func (c *Config) UnpackWithoutOptions(to interface{}) error
- type ConfigUnpacker
- type Error
- type FloatUnpacker
- type Initializer
- type IntUnpacker
- type Meta
- type Option
- type SecretString
- func (k SecretString) Get() string
- func (k SecretString) GoString() string
- func (k SecretString) MarshalJSON() ([]byte, error)
- func (k SecretString) MarshalYAML() (interface{}, error)
- func (k SecretString) String() string
- func (k *SecretString) UnmarshalJSON(src []byte) error
- func (k *SecretString) UnmarshalYAML(unmarshal func(interface{}) error) error
- type StringUnpacker
- type UintUnpacker
- type Unpacker
- type Validator
- type ValidatorCallback
Constants ¶
const ( SecretStringDelimiter = "-->" SecretShadowText = "******" )
var ErrMalformed = errors.New("secret string malformed")
Variables ¶
var ( ErrMissing = errors.New("missing field") ErrNoParse = errors.New("parsing dynamic configs is disabled") ErrCyclicReference = errors.New("cyclic reference detected") ErrDuplicateValidator = errors.New("validator already registered") ErrTypeNoArray = errors.New("field is no array") ErrTypeMismatch = errors.New("type mismatch") ErrKeyTypeNotString = errors.New("key must be a string") ErrIndexOutOfRange = errors.New("out of range index") ErrPointerRequired = errors.New("pointer required for unpacking configurations") ErrArraySizeMismatch = errors.New("Array size mismatch") ErrExpectedObject = errors.New("expected object") ErrNilConfig = errors.New("config is nil") ErrNilValue = errors.New("nil value is invalid") ErrTODO = errors.New("TODO - implement me") ErrDuplicateKey = errors.New("duplicate key") ErrOverflow = errors.New("integer overflow") ErrNegative = errors.New("negative value") ErrZeroValue = errors.New("zero value") ErrRequired = errors.New("missing required field") ErrEmpty = errors.New("empty field") ErrArrayEmpty = errors.New("empty array") ErrMapEmpty = errors.New("empty map") ErrRegexEmpty = errors.New("regex value is not set") ErrStringEmpty = errors.New("string value is not set") )
Error Reasons
var ( ErrConfig = errors.New("Configuration error") ErrImplementation = errors.New("Implementation error") ErrUnknown = errors.New("Unspecified") )
Error Classes
var ( // ReplaceValues option configures all merging and unpacking operations to // replace old dictionaries and arrays while merging. Value merging can be // overwritten in unpack by using struct tags. ReplaceValues = makeOptValueHandling(cfgReplaceValue) // AppendValues option configures all merging and unpacking operations to // merge dictionaries and append arrays to existing arrays while merging. // Value merging can be overwritten in unpack by using struct tags. AppendValues = makeOptValueHandling(cfgArrAppend) // PrependValues option configures all merging and unpacking operations to // merge dictionaries and prepend arrays to existing arrays while merging. // Value merging can be overwritten in unpack by using struct tags. PrependValues = makeOptValueHandling(cfgArrPrepend) )
var ( // FieldMergeValues option configures all merging and unpacking operations to use // the default merging behavior for the specified field. This overrides the any struct // tags during unpack for the field. Nested field names can be defined using dot // notation. FieldMergeValues = makeFieldOptValueHandling(cfgMergeValues) // FieldReplaceValues option configures all merging and unpacking operations to // replace old dictionaries and arrays while merging for the specified field. This // overrides the any struct tags during unpack for the field. Nested field names // can be defined using dot notation. FieldReplaceValues = makeFieldOptValueHandling(cfgReplaceValue) // FieldAppendValues option configures all merging and unpacking operations to // merge dictionaries and append arrays to existing arrays while merging for the // specified field. This overrides the any struct tags during unpack for the field. // Nested field names can be defined using dot notation. FieldAppendValues = makeFieldOptValueHandling(cfgArrAppend) // FieldPrependValues option configures all merging and unpacking operations to // merge dictionaries and prepend arrays to existing arrays while merging for the // specified field. This overrides the any struct tags during unpack for the field. // Nested field names can be defined using dot notation. FieldPrependValues = makeFieldOptValueHandling(cfgArrPrepend) )
Functions ¶
func DecodeSecretString ¶
func RegisterValidator ¶
func RegisterValidator(name string, cb ValidatorCallback) error
RegisterValidator adds a new validator option to the "validate" struct tag. The callback will be executed when unpacking into a struct field.
Types ¶
type BoolUnpacker ¶
BoolUnpacker interface specializes the Unpacker interface by casting values to bool when calling Unpack.
type Config ¶
type Config struct {
// contains filtered or unexported fields
}
Config object to store hierarchical configurations into. Config can be both a dictionary and a list holding primitive values. Primitive values can be booleans, integers, float point numbers and strings.
Config provides a low level interface for setting and getting settings via SetBool, SetInt, SetUint, SetFloat, SetString, SetChild, Bool, Int, Uint, Float, String, and Child.
A more user-friendly high level interface is provided via Unpack and Merge.
func MustNewFrom ¶
MustNewFrom creates a new config object normalizing and copying from into the new Config object. MustNewFrom uses Merge to copy from.
MustNewFrom supports the options: PathSep, MetaData, StructTag, VarExp
func NewFrom ¶
NewFrom creates a new config object normalizing and copying from into the new Config object. NewFrom uses Merge to copy from.
NewFrom supports the options: PathSep, MetaData, StructTag, VarExp
func (*Config) Bool ¶
Bool reads a boolean setting returning an error if the setting has no boolean value.
The setting path is constructed from name and idx. If name is set and idx is -1, only the name is used to access the setting by name. If name is empty, idx must be >= 0, assuming the Config is a list. If both name and idx are set, the name must point to a list. The number of entries in a named list can be read using CountField.
Bool supports the options: PathSep, Env, Resolve, ResolveEnv
func (*Config) Child ¶
Child returns a child configuration or an error if the setting requested is a primitive value only.
The setting path is constructed from name and idx. If name is set and idx is -1, only the name is used to access the setting by name. If name is empty, idx must be >= 0, assuming the Config is a list. If both name and idx are set, the name must point to a list. The number of entries in a named list can be read using CountField.
Child supports the options: PathSep, Env, Resolve, ResolveEnv
func (*Config) CountField ¶
CountField returns number of entries in a table or 1 if entry is a primitive value. Primitives settings can be handled like a list with 1 entry.
If name is empty, the total number of top-level settings is returned.
CountField supports the options: PathSep, Env, Resolve, ResolveEnv
func (*Config) FlattenedKeys ¶
FlattenedKeys return a sorted flattened views of the set keys in the configuration
func (*Config) Float ¶
Float reads a float64 value returning an error if the setting is not a float value or the primitive value is not convertible to float.
The setting path is constructed from name and idx. If name is set and idx is -1, only the name is used to access the setting by name. If name is empty, idx must be >= 0, assuming the Config is a list. If both name and idx are set, the name must point to a list. The number of entries in a named list can be read using CountField.
Float supports the options: PathSep, Env, Resolve, ResolveEnv
func (*Config) Has ¶
Has checks if a field by the given path+idx configuration exists. Has returns an error if the path can not be resolved because a primitive value is found in the middle of the traversal.
func (*Config) Int ¶
Int reads an int64 value returning an error if the setting is not integer value, the primitive value is not convertible to int or a conversion would create an integer overflow.
The setting path is constructed from name and idx. If name is set and idx is -1, only the name is used to access the setting by name. If name is empty, idx must be >= 0, assuming the Config is a list. If both name and idx are set, the name must point to a list. The number of entries in a named list can be read using CountField.
Int supports the options: PathSep, Env, Resolve, ResolveEnv
func (*Config) Merge ¶
Merge a map, a slice, a struct or another Config object into c.
Merge traverses the value from recursively copying all values into a hierarchy of Config objects plus primitives into c.
Merge supports the options: PathSep, MetaData, StructTag, VarExp, ReplaceValues, AppendValues, PrependValues
Merge uses the type-dependent default encodings:
- Boolean values are encoded as booleans.
- Integer are encoded as int64 values, unsigned integer values as uint64 and floats as float64 values.
- Strings are copied into string values. If the VarExp is set, string fields will be parsed into variable expansion expressions. The expression can reference any other setting by absolute name.
- Array and slices are copied into new Config objects with index accessors only.
- Struct values and maps with key type string are encoded as Config objects with named field accessors.
- Config objects will be copied and added to the current hierarchy.
The `config` struct tag (configurable via StructTag option) can be used to set the field name and enable additional merging settings per field:
// field appears in Config as key "myName" Field int `config:"myName"` // field appears in sub-Config "mySub" as key "myName" (requires PathSep(".")) Field int `config:"mySub.myName"` // field is processed as if keys are part of outer struct (type can be a // struct, a slice, an array, a map or of type *Config) Field map[string]interface{} `config:",inline"` // field is ignored by Merge Field string `config:",ignore"`
Returns an error if merging fails to normalize and validate the from value. If duplicate setting names are detected in the input, merging fails as well.
Config cannot represent cyclic structures and Merge does not handle them well. Passing cyclic structures to Merge will result in an infinite recursive loop.
func (*Config) Parent ¶
Parent returns the parent configuration or nil if c is already a root Configuration.
func (*Config) Path ¶
Path gets the absolute path of c separated by sep. If c is a root-Config an empty string will be returned.
func (*Config) PathOf ¶
PathOf gets the absolute path of a potential setting field in c with name separated by sep.
func (*Config) Remove ¶
Remove removes a setting from the config. If the configuration references another configuration namespace, then the setting will be removed from the linked reference. Remove returns true if the setting was removed. If the path can't be resolved (e.g. due to type mismatch) Remove will return an error.
Settings can be created on Unpack via Env, Resolve, and ResolveEnv. Settings generated dynamically on Unpack can not be removed. Remove ignores any configured environments and will return an error if a value can not be removed for this reason.
The setting path is constructed from name and idx. If name is set and idx is -1, only the name is used to access the setting by name. If name is empty, idx must be >= 0, assuming the Config is a list. If both name and idx are set, the name must point to a list.
Remove supports the options: PathSep
func (*Config) SetBool ¶
SetBool sets a boolean primitive value. An error is returned if the new name is invalid.
The setting path is constructed from name and idx. If name is set and idx is -1, only the name is used to access the setting by name. If name is empty, idx must be >= 0, assuming the Config is a list. If both name and idx are set, the name must point to a list. The number of entries in a named list can be read using CountField.
SetBool supports the options: PathSep, MetaData
func (*Config) SetChild ¶
SetChild adds a sub-configuration. An error is returned if the name is invalid.
The setting path is constructed from name and idx. If name is set and idx is -1, only the name is used to access the setting by name. If name is empty, idx must be >= 0, assuming the Config is a list. If both name and idx are set, the name must point to a list. The number of entries in a named list can be read using CountField.
SetChild supports the options: PathSep, MetaData
func (*Config) SetFloat ¶
SetFloat sets an floating point primitive value. An error is returned if the name is invalid.
The setting path is constructed from name and idx. If name is set and idx is -1, only the name is used to access the setting by name. If name is empty, idx must be >= 0, assuming the Config is a list. If both name and idx are set, the name must point to a list. The number of entries in a named list can be read using CountField.
SetFloat supports the options: PathSep, MetaData
func (*Config) SetInt ¶
SetInt sets an integer primitive value. An error is returned if the new name is invalid.
The setting path is constructed from name and idx. If name is set and idx is -1, only the name is used to access the setting by name. If name is empty, idx must be >= 0, assuming the Config is a list. If both name and idx are set, the name must point to a list. The number of entries in a named list can be read using CountField.
SetInt supports the options: PathSep, MetaData
func (*Config) SetString ¶
SetString sets string value. An error is returned if the name is invalid.
The setting path is constructed from name and idx. If name is set and idx is -1, only the name is used to access the setting by name. If name is empty, idx must be >= 0, assuming the Config is a list. If both name and idx are set, the name must point to a list. The number of entries in a named list can be read using CountField.
SetString supports the options: PathSep, MetaData
func (*Config) SetUint ¶
SetUint sets an unsigned integer primitive value. An error is returned if the name is invalid.
The setting path is constructed from name and idx. If name is set and idx is -1, only the name is used to access the setting by name. If name is empty, idx must be >= 0, assuming the Config is a list. If both name and idx are set, the name must point to a list. The number of entries in a named list can be read using CountField.
SetUint supports the options: PathSep, MetaData
func (*Config) String ¶
Strings reads a string setting returning an error if the setting has no string or primitive value convertible to string.
The setting path is constructed from name and idx. If name is set and idx is -1, only the name is used to access the setting by name. If name is empty, idx must be >= 0, assuming the Config is a list. If both name and idx are set, the name must point to a list. The number of entries in a named list can be read using CountField.
String supports the options: PathSep, Env, Resolve, ResolveEnv
func (*Config) Uint ¶
Uint reads an uint64 value returning an error if the setting is not unsigned value, the primitive value is not convertible to uint64 or a conversion would create an integer overflow.
The setting path is constructed from name and idx. If name is set and idx is -1, only the name is used to access the setting by name. If name is empty, idx must be >= 0, assuming the Config is a list. If both name and idx are set, the name must point to a list. The number of entries in a named list can be read using CountField.
Uint supports the options: PathSep, Env, Resolve, ResolveEnv
func (*Config) Unpack ¶
Unpack unpacks c into a struct, a map, or a slice allocating maps, slices, and pointers as necessary.
Unpack supports the options: PathSep, StructTag, ValidatorTag, Env, Resolve, ResolveEnv, ReplaceValues, AppendValues, PrependValues.
When unpacking into a value, Unpack first will try to call Unpack if the value implements the Unpacker interface. Otherwise, Unpack tries to convert the internal value into the target type:
# Primitive types bool: requires setting of type bool or string which parses into a boolean value (true, false, on, off) int(8, 16, 32, 64): requires any number type convertible to int or a string parsing to int. Fails if the target value would overflow. uint(8, 16, 32, 64): requires any number type convertible to int or a string parsing to int. Fails if the target value is negative or would overflow. float(32, 64): requires any number type convertible to float or a string parsing to float. Fails if the target value is negative or would overflow. string: requires any primitive value which is serialized into a string. # Special types: time.Duration: requires a number setting converted to seconds or a string parsed into time.Duration via time.ParseDuration. *regexp.Regexp: requires a string being compiled into a regular expression using regexp.Compile. *Config: requires a Config object to be stored by pointer into the target value. Can be used to capture a sub-Config without interpreting the settings yet. # Arrays/Slices: Requires a Config object with indexed entries. Named entries will not be unpacked into the Array/Slice. Primitive values will be handled like arrays of length 1. # Map Requires a Config object with all named top-level entries being unpacked into the map. # Struct Requires a Config object. All named values in the Config object will be unpacked into the struct its fields, if the name is available in the struct. A field its name is set using the `config` struct tag (configured by StructTag) If tag is missing or no field name is configured in the tag, the field name itself will be used. If the tag sets the `,ignore` flag, the field will not be overwritten. If the tag sets the `,inline` or `,squash` flag, Unpack will apply the current configuration namespace to the fields. If the tag option `replace` is configured, arrays and *ucfg.Config convertible fields are replaced by the new values. If the tag options `append` or `prepend` is used, arrays will be merged by appending/prepending the new array contents. The struct tag options `replace`, `append`, and `prepend` overwrites the global value merging strategy (e.g. ReplaceValues, AppendValues, ...) for all sub-fields.
When unpacking into a map, primitive, or struct Unpack will call InitDefaults if the type implements the Initializer interface. The Initializer interface is not supported on arrays or slices. InitDefaults is initialized top-down, meaning that if struct contains a map, struct, or primitive that also implements the Initializer interface the contained type will be initialized after the struct that contains it. (e.g. if we have type A struct { B B }, with both A, and B implementing InitDefaults, then A.InitDefaults is called before B.InitDefaults). In the case that a struct contains a pointer to a type that implements the Initializer interface and the configuration doesn't contain a value for that field then the pointer will not be initialized and InitDefaults will not be called.
Fields available in a struct or a map, but not in the Config object, will not be touched by Unpack unless they are initialized from InitDefaults. Those values will be validated using the same rules below just as if the values came from the configuration. This gives the requirement that pre-filled in values or defaults must also validate.
Type aliases like "type myTypeAlias T" are unpacked using Unpack if the alias implements the Unpacker interface. Otherwise unpacking rules for type T will be used.
When unpacking a value, the Validate method will be called if the value implements the Validator interface. Unpacking a struct field the validator options will be applied to the unpacked value as well.
Struct field validators are set using the `validate` tag (configurable by ValidatorTag). Default validators options are:
required: check value is set and not empty nonzero: check numeric value != 0 or string/slice not being empty positive: check numeric value >= 0 min=<value>: check numeric value >= <value>. If target type is time.Duration, <value> can be a duration. max=<value>: check numeric value <= <value>. If target type is time.Duration, <value> can be a duration.
If a config value is not the convertible to the target type, or overflows the target type, Unpack will abort immediately and return the appropriate error.
If validator tags or validation provided by Validate or Unmarshal fails, Unpack will abort immediately and return the validate error.
When unpacking into an interface{} value, Unpack will store a value of one of these types in the value:
bool for boolean values int64 for signed integer values uint64 for unsigned integer values float64 for floating point values string for string values []interface{} for list-only Config objects map[string]interface{} for Config objects nil for pointers if key has a nil value
func (*Config) UnpackWithoutOptions ¶
UnpackWithoutOptions method calls the Unpack method without any options provided.
type ConfigUnpacker ¶
ConfigUnpacker interface specializes the Unpacker interface by passing the the *Config object directly instead of transforming the *Config object into map[string]interface{}.
type Error ¶
type Error interface { error // error class, one of ErrConfig, ErrImplementation, ErrUnknown Class() error // The internal reason error code like ErrMissing, ErrRequired, // ErrTypeMismatch and others. Reason() error // The error message. Message() string // [optional] path of config element error occurred for Path() string // [optional] stack trace Trace() string }
Error type returned by all public functions in go-ucfg.
type FloatUnpacker ¶
FloatUnpacker interface specializes the Unpacker interface by casting values to float64 when calling Unpack.
type Initializer ¶
type Initializer interface {
InitDefaults()
}
Initializer interface provides initialization of default values support to Unpack. The InitDefaults method will be executed for any type passed directly or indirectly to Unpack.
type IntUnpacker ¶
IntUnpacker interface specializes the Unpacker interface by casting values to int64 when calling Unpack.
type Option ¶
type Option func(*options)
Option type implementing additional options to be passed to go-ucfg library functions.
var NoResolve Option = doNoResolve
NoResolve option sets do not to resolve variables.
var ResolveEnv Option = doResolveEnv
ResolveEnv option adds a look up callback looking up values in the available OS environment variables.
var ResolveNOOP Option = doResolveNOOP
ResolveNOOP option add a resolver that will not search the value but instead will return the provided key wrap with the field reference syntax. This is useful if you don't to expose values from envionment variable or other resolvers.
Example: "mysecret" => ${mysecret}"
var ResolveRef Option = doResolveRef
ResolveRef option enables support for variable resolve reference from parent config.
var VarExp Option = doVarExp
VarExp option enables support for variable expansion. Resolve and Env options will only be effective if VarExp is set.
func DefaultParseConfig ¶
DefaultParseConfig option sets the default parse config used to parse dyn value if it is not empty
func Env ¶
Env option adds another configuration for variable expansion to be used, if the path to look up does not exist in the actual configuration. Env can be used multiple times in order to add more lookup environments.
func MetaData ¶
MetaData option passes additional metadata (currently only source of the configuration) to be stored internally (e.g. for error reporting).
func PathSep ¶
PathSep sets the path separator used to split up names into a tree like hierarchy. If PathSep is not set, field names will not be split.
func Resolve ¶
Resolve option adds a callback used by variable name expansion. The callback will be called if a variable can not be resolved from within the actual configuration or any of its environments.
func StructTag ¶
StructTag option sets the struct tag name to use for looking up field names and options in `Unpack` and `Merge`. The default struct tag in `config`.
func ValidatorTag ¶
ValidatorTag option sets the struct tag name used to set validators on struct fields in `Unpack`. The default struct tag in `validate`.
type SecretString ¶
type SecretString string
func EncodeToSecretString ¶
func EncodeToSecretString(raw string, value string) SecretString
func (SecretString) Get ¶
func (k SecretString) Get() string
func (SecretString) GoString ¶
func (k SecretString) GoString() string
func (SecretString) MarshalJSON ¶
func (k SecretString) MarshalJSON() ([]byte, error)
func (SecretString) MarshalYAML ¶
func (k SecretString) MarshalYAML() (interface{}, error)
func (SecretString) String ¶
func (k SecretString) String() string
func (*SecretString) UnmarshalJSON ¶
func (k *SecretString) UnmarshalJSON(src []byte) error
func (*SecretString) UnmarshalYAML ¶
func (k *SecretString) UnmarshalYAML(unmarshal func(interface{}) error) error
type StringUnpacker ¶
StringUnpacker interface specializes the Unpacker interface by casting values to string when calling Unpack.
type UintUnpacker ¶
UintUnpacker interface specializes the Unpacker interface by casting values to uint64 when calling Unpack.
type Unpacker ¶
type Unpacker interface { // Unpack is called if a setting of field has a type implementing Unpacker. // // The interface{} value passed to Unpack can be of type: bool, int64, uint64, // float64, string, []interface{} or map[string]interface{}. Unpack(interface{}) error }
Unpacker type used by Unpack to allow types to implement custom configuration unpacking.
type Validator ¶
type Validator interface {
Validate() error
}
Validator interface provides additional validation support to Unpack. The Validate method will be executed for any type passed directly or indirectly to Unpack.
If Validate fails with an error message, Unpack will add some context - like setting being accessed and file setting was read from - to the error message before returning the actual error.
type ValidatorCallback ¶
ValidatorCallback is the type of optional validator tags to be registered via RegisterValidator.