EnvConfig is a go library which enables you to populate a struct according to the process environment. It infers environment variables names according to struct field names.
It fully supports complex structures involving maps, arrays and slices, also it deals with allocation.
However, at the moment it doesn't support type loops, ie (TypeA => TypeB => TypeA ...)
Here's a basic example.
package example
import (
"github.com/jlevesy/envconfig/setter"
"github.com/jlevesy/envconfig"
)
const (
AppPrefix = "TEST"
Separator = "_"
)
//Configuration is a struct which contains all differents type to field
type Configuration struct {
IntField int
StringField string
PointerField *PointerSubConfiguration
}
//PointerSubConfiguration is a SubStructure Configuration
type PointerSubConfiguration struct {
BoolField bool
FloatField float64
}
Let's initialize it:
func main() {
config := &Configuration{}
// load your structure
if err := envconfig.New(AppPrefix, Separator).Load(config); err != nil {
fmt.Println("Failed to load config, got: ", err)
os.Exit(1)
}
// Use your filled structure
}
Now if I run...
TEST_INT_FIELD=10 TEST_POINTER_FIELD_BOOL_FIELD=1 go run main.go
... config.IntField
will be set to 10, and config.PointerField.BoolField
to
true !
And that's pretty much it ! If you need more details there is a detailed example.
It can be initalized like this :
prefix := "FOO"
speparator := "/"
env := envconfig.New(prefix, separator)
It takes two arguments:
- A prefix used in order to format environment variables names to fetch, if left blank, no prefix will be applied to environment variables
- A separator string, if left blank it will default to the "_" string
Another constructor is available
env := envconfig.NewWithSettersAndDepth(prefix, separator, setters, maxDepth)
It adds two more arguments
- A setter collection which is a
map[reflect.Type]setter.Setter
representing all types envConfig can write to. - A maxdepth, setting a hard limit on structure depth to avoid type loops.
envconfig.New(prefix, separator)
, is equivalent to envconfig.NewWithSettersAndDepth(prefix, separator, setter.LoadBasicTypes(), 10)
Environment variable names are structured like this:
[PREFIX][SEP][MY][SEP][FIELD][SEP][NAME]
Field names are split into by words according to camelCase, we rely on github.com/fatih/camelcase to do this.
For instance if prefix is "MyApp" and separator is the '_' rune we'll have the following mapping:
type AppStruct struct {
MyStringField string // => MYAPP_MY_STRING_FIELD
MyIntField int // => MYAPP_MY_INT_FIELD
}
Embedded structures are supported, and environment variable name generation for a field will have exact same behaviour than normal struct field.
For instance if we keep our previous example confifuration, we'll obtain the following mapping:
type CommonConfig struct {
CommonString string // => MYAPP_COMMON_STRING
}
type AppConfig struct {
CommonConfig
}
Nested structures are also supported, both by pointer and values. However fields names are going to be prefixed with the field name referencing the nested structure:
type PtrNestedConfig struct {
AnArgument string // => MY_APP_FOO_AN_ARGUMENT
}
type ValueNestedConfig struct {
AnotherArgument string // => MY_APP_BAR_ANOTHER_ARGUMENT
}
type AppConfig struct {
Foo **PtrNestedConfig // => Double indirection because why not ?
Bar ValueNestedConfig
}
You can also use pointer to values too in your config structs, those fields are going to be mapped exactly as a value.
type AppConfig struct {
Groot *int32 // => MYAPP_GROOT
}
You can affect values into array and slices using environment variables. Index affectation is not guaranteed, but ordering is.
type NestedAppConfig struct {
BoolValue bool // => MY_APP_BAR_<INT_INDEX>_BOOL_VALUE
}
type AppConfig struct {
Foo []string // => MY_APP_FOO_<INT_INDEX>
Bar []*NestedAppConfig
}
You can affect values into maps, just like arrays and slices, however key type must be supported by the setter colllection.
type NestedAppConfig struct {
BoolValue bool // => MY_APP_BAR_<KEY>_BOOL_VALUE
}
type AppConfig struct {
Foo map[int]string // => MY_APP_FOO_<KEY>
Bar map[float64]*NestedAppConfig
}
Sometimes you might want to valuate structs using a smarter string parsing instead of definining multiple environment variables.
If you tag your struct field with the envconfig:"noexpand"
tag,
envconfig will try to assign the given string value to the struct field
and rely on a custom Setter
to deal with parsing and assignment.
It supports structs, slices and maps
A small example to illustrate, let's say I want to set a slice of string as an environment variable
// main.go
import(
"reflect"
"strings"
"github.com/jlevesy/envconfig"
"github.com/jlevesy/envconfig/setter"
)
type ConfigStruct {
// Define a field Repos with the right struct tag
Repos []string `envconfig:"noexpand"`
}
// Setter for []string in our app
// it splits given string according to the comma character
func sliceOfStringSetter(strValue string, value reflect.Value) error {
value.Set(reflect.ValueOf(strings.Split(strValue, ",")))
return nil
}
func main() {
config := &ConfigStruct{}
setters := setter.LoadBasicTypes()
// define your setterFunc as setter for the type []string
setters[reflect.TypeOf([]string{})] = setter.SetterFunc(sliceOfStringSetter)
// Now load your configuration using your setters collection
if err := envconfig.NewWithSettersAndDepth("APP", "_", setters, 10).Load(config); err != nil {
// Fail gracefuly
}
// Do something awesome with your config [...]
}
If I run APP_REPOS="foo,bar,buz" go run main.go
loaded config will
have the value {Items:["foo","bar","buz"]}
EnvConfig depends on a setter collection representing all types it can write to.
A Setter is defined by the following interface.
type Setter interface {
Set(strValue string, val reflect.Value) error
}
If you need to support different types, for instance an IP address, feel free to
define your very own Setter
or SetterFunc
, and add it to your setter
collection at initialization.
Be careful however, because setting a invalid value using the reflect
library might result in a panic !
- Control structure expanding using struct tags
- Support custom environment variable names using tags
- Better structure loop detection
Of course, any suggestions are welcome ! :)
- Fork it!
- Create your feature branch:
git checkout -b my-new-feature
- Commit your changes:
git commit -am 'Add some feature'
- Push to the branch:
git push origin my-new-feature
- Submit a pull request :D