configloader is a go library for loading configuration from multiple sources.
The github.com/pastdev/pkg/config package provides the core configuration loading library.
The library itself is very simple in that it has a single interface (SourceLoader) that can be implemented for any type of source.
These source loader instances are aggregated together in a Sources object that can then be used to load and merge multiple sources together.
Subsequent sources will merge their values over the top of any existing values so the latest defined wins.
sources := config.Sources[AppConfig]{
config.FileSource[AppConfig]{Path: "~/.config/app.yml"},
config.DirSource[AppConfig]{Path: "~/.config/app.d"},
}
sources.Load(&cfg)See the example or tests for more use cases.
By default, YamlUnmarshal is used.
However, you can replace that with a custom unmarshaler if you would like:
sources := config.Sources[AppConfig]{
config.FileSource[map[any]any]{
Path: "~/.config/configloader.tmpl.d",
Unmarshal: func(b []byte, cfg *map[any]any) error {
return json.Unmarshal(b, cfg)
},
},
}You can also configure your source loaders to pre-process config file values with the go templating engine:
sources := config.Sources[AppConfig]{
config.FileSource[AppConfig]{
Path: "/etc/configloader.tmpl.yml",
Unmarshal: config.
YamlValueTemplateUnmarshal[AppConfig](
config.NewTemplate(config.DefaultFuncMap()))
},
}The config.DefaultFuncMap() contains utility functions for accessing secrets from various password managers (ie: lastpass, bitwarden).
This map can be added to, or replaced.
To use the bitwarden template functions, you need to install the rbw client.
The template functions assume you have an active session (ie: rbw unlock) from which it will obtain the secrets.
To use the lastpass template functions, you need to install the lastpass-cli client.
The template functions assume you have an active session (ie: lpass login <USER>) from which it will obtain the secrets.
This library uses zerolog for logging.
The Logger can be set by consumers as follows:
import(
"github.com/pastdev/configloader/pkg/log"
)
...
log.Logger = zerolog.New(os.Stderr).Level(zerolog.TraceLevel).With().Timestamp().Logger()The github.com/pastdev/pkg/cobra package provides CLI integration with cobra.
There is 1 mandatory, and 2 optional integration points.
First you need to define your ConfigLoader object:
cfgldr := cobraconfig.ConfigLoader[map[any]any]{
DefaultSources: config.Sources[map[any]any]{
config.FileSource[map[any]any]{Path: "/etc/configloader.yml"},
config.DirSource[map[any]any]{Path: "/etc/configloader.d"},
config.FileSource[map[any]any]{Path: "~/.config/configloader.yml"},
config.DirSource[map[any]any]{Path: "~/.config/configloader.d"},
},
}Optionally, you can use flags to allow your user to replace the DefaultSources:
cfg.PersistentFlags(&root).FileSourceVar(
config.YamlUnmarshal[map[any]any](),
"config",
"location of one or more config files")
cfg.PersistentFlags(&root).DirSourceVar(
config.YamlUnmarshal[map[any]any](),
"config-dir",
"location of one or more config directories")And add a config subcommand to your root command for printing out the configuration:
cfg.AddSubCommandTo(&root)Or a use additional options when adding the subcommand:
cfgldr.AddSubCommandTo(
&root,
cobraconfig.WithConfigCommandOutput(
"json",
func(w io.Writer, cfg *map[any]any) error {
jsonmap := map[string]any{}
for k, v := range *cfg {
jsonmap[fmt.Sprintf("%s", k)] = v
}
err := json.NewEncoder(w).Encode(jsonmap)
if err != nil {
return fmt.Errorf("format json: %w", err)
}
return nil
},
),
cobraconfig.WithConfigCommandSilenceUsage[map[any]any](true))Then you can pass the the configloader object to any subcommands and simply call the .Config() method to load and access the config object:
root.AddCommand(fooCmd(&cfgldr))
...
func fooCmd(cfgldr *cobraconfig.ConfigLoader[map[any]any]) *cobra.Command {
return &cobra.Command{
Use: "foo",
Short: `An example subcommand for how to use configloader to show the value of foo.`,
RunE: func(_ *cobra.Command, _ []string) error {
cfg, err := cfgldr.Config()
if err != nil {
return fmt.Errorf("get config: %w", err)
}
fmt.Printf("foo is [%s]", (*cfg)["foo"])
return nil
},
}
}