package configuration import ( "fmt" "github.com/knadh/koanf" "github.com/mitchellh/mapstructure" "github.com/authelia/authelia/v4/internal/configuration/schema" ) // Load the configuration given the provided options and sources. func Load(val *schema.StructValidator, sources ...Source) (keys []string, configuration *schema.Configuration, err error) { configuration = &schema.Configuration{} keys, err = LoadAdvanced(val, "", configuration, sources...) return keys, configuration, err } // LoadAdvanced is intended to give more flexibility over loading a particular path to a specific interface. func LoadAdvanced(val *schema.StructValidator, path string, result interface{}, sources ...Source) (keys []string, err error) { if val == nil { return keys, errNoValidator } ko := koanf.NewWithConf(koanf.Conf{ Delim: constDelimiter, StrictMerge: false, }) if err = loadSources(ko, val, sources...); err != nil { return ko.Keys(), err } var final *koanf.Koanf if final, err = koanfRemapKeys(val, ko, deprecations); err != nil { return koanfGetKeys(ko), err } unmarshal(final, val, path, result) return koanfGetKeys(final), nil } func mapHasKey(k string, m map[string]interface{}) bool { if _, ok := m[k]; ok { return true } return false } func unmarshal(ko *koanf.Koanf, val *schema.StructValidator, path string, o interface{}) { c := koanf.UnmarshalConf{ DecoderConfig: &mapstructure.DecoderConfig{ DecodeHook: mapstructure.ComposeDecodeHookFunc( mapstructure.StringToSliceHookFunc(","), StringToMailAddressHookFunc(), StringToURLHookFunc(), StringToRegexpHookFunc(), StringToAddressHookFunc(), StringToX509CertificateHookFunc(), StringToX509CertificateChainHookFunc(), StringToPrivateKeyHookFunc(), ToTimeDurationHookFunc(), ), Metadata: nil, Result: o, WeaklyTypedInput: true, }, } if err := ko.UnmarshalWithConf(path, o, c); err != nil { val.Push(fmt.Errorf("error occurred during unmarshalling configuration: %w", err)) } } func loadSources(ko *koanf.Koanf, val *schema.StructValidator, sources ...Source) (err error) { if len(sources) == 0 { return errNoSources } for _, source := range sources { err := source.Load(val) if err != nil { val.Push(fmt.Errorf("failed to load configuration from %s source: %+v", source.Name(), err)) continue } err = source.Merge(ko, val) if err != nil { val.Push(fmt.Errorf("failed to merge configuration from %s source: %+v", source.Name(), err)) continue } } return nil }