175 lignes
4,7 Kio
Go
175 lignes
4,7 Kio
Go
|
package configor
|
||
|
|
||
|
import (
|
||
|
"encoding/json"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"io/ioutil"
|
||
|
"os"
|
||
|
"path"
|
||
|
"reflect"
|
||
|
"strings"
|
||
|
|
||
|
"github.com/BurntSushi/toml"
|
||
|
yaml "gopkg.in/yaml.v1"
|
||
|
)
|
||
|
|
||
|
func (configor *Configor) getENVPrefix(config interface{}) string {
|
||
|
if configor.Config.ENVPrefix == "" {
|
||
|
if prefix := os.Getenv("CONFIGOR_ENV_PREFIX"); prefix != "" {
|
||
|
return prefix
|
||
|
}
|
||
|
return "Configor"
|
||
|
}
|
||
|
return configor.Config.ENVPrefix
|
||
|
}
|
||
|
|
||
|
func getConfigurationFileWithENVPrefix(file, env string) (string, error) {
|
||
|
var (
|
||
|
envFile string
|
||
|
extname = path.Ext(file)
|
||
|
)
|
||
|
|
||
|
if extname == "" {
|
||
|
envFile = fmt.Sprintf("%v.%v", file, env)
|
||
|
} else {
|
||
|
envFile = fmt.Sprintf("%v.%v%v", strings.TrimSuffix(file, extname), env, extname)
|
||
|
}
|
||
|
|
||
|
if fileInfo, err := os.Stat(envFile); err == nil && fileInfo.Mode().IsRegular() {
|
||
|
return envFile, nil
|
||
|
}
|
||
|
return "", fmt.Errorf("failed to find file %v", file)
|
||
|
}
|
||
|
|
||
|
func (configor *Configor) getConfigurationFiles(files ...string) []string {
|
||
|
var results []string
|
||
|
|
||
|
for i := len(files) - 1; i >= 0; i-- {
|
||
|
foundFile := false
|
||
|
file := files[i]
|
||
|
|
||
|
// check configuration
|
||
|
if fileInfo, err := os.Stat(file); err == nil && fileInfo.Mode().IsRegular() {
|
||
|
foundFile = true
|
||
|
results = append(results, file)
|
||
|
}
|
||
|
|
||
|
// check configuration with env
|
||
|
if file, err := getConfigurationFileWithENVPrefix(file, configor.GetEnvironment()); err == nil {
|
||
|
foundFile = true
|
||
|
results = append(results, file)
|
||
|
}
|
||
|
|
||
|
// check example configuration
|
||
|
if !foundFile {
|
||
|
if example, err := getConfigurationFileWithENVPrefix(file, "example"); err == nil {
|
||
|
fmt.Printf("Failed to find configuration %v, using example file %v\n", file, example)
|
||
|
results = append(results, example)
|
||
|
} else {
|
||
|
fmt.Printf("Failed to find configuration %v\n", file)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return results
|
||
|
}
|
||
|
|
||
|
func processFile(config interface{}, file string) error {
|
||
|
data, err := ioutil.ReadFile(file)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
switch {
|
||
|
case strings.HasSuffix(file, ".yaml") || strings.HasSuffix(file, ".yml"):
|
||
|
return yaml.Unmarshal(data, config)
|
||
|
case strings.HasSuffix(file, ".toml"):
|
||
|
return toml.Unmarshal(data, config)
|
||
|
case strings.HasSuffix(file, ".json"):
|
||
|
return json.Unmarshal(data, config)
|
||
|
default:
|
||
|
if toml.Unmarshal(data, config) != nil {
|
||
|
if json.Unmarshal(data, config) != nil {
|
||
|
if yaml.Unmarshal(data, config) != nil {
|
||
|
return errors.New("failed to decode config")
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func getPrefixForStruct(prefixes []string, fieldStruct *reflect.StructField) []string {
|
||
|
if fieldStruct.Anonymous && fieldStruct.Tag.Get("anonymous") == "true" {
|
||
|
return prefixes
|
||
|
}
|
||
|
return append(prefixes, fieldStruct.Name)
|
||
|
}
|
||
|
|
||
|
func processTags(config interface{}, prefixes ...string) error {
|
||
|
configValue := reflect.Indirect(reflect.ValueOf(config))
|
||
|
if configValue.Kind() != reflect.Struct {
|
||
|
return errors.New("invalid config, should be struct")
|
||
|
}
|
||
|
|
||
|
configType := configValue.Type()
|
||
|
for i := 0; i < configType.NumField(); i++ {
|
||
|
var (
|
||
|
envNames []string
|
||
|
fieldStruct = configType.Field(i)
|
||
|
field = configValue.Field(i)
|
||
|
envName = fieldStruct.Tag.Get("env") // read configuration from shell env
|
||
|
)
|
||
|
|
||
|
if envName == "" {
|
||
|
envNames = append(envNames, strings.Join(append(prefixes, fieldStruct.Name), "_")) // Configor_DB_Name
|
||
|
envNames = append(envNames, strings.ToUpper(strings.Join(append(prefixes, fieldStruct.Name), "_"))) // CONFIGOR_DB_NAME
|
||
|
} else {
|
||
|
envNames = []string{envName}
|
||
|
}
|
||
|
|
||
|
// Load From Shell ENV
|
||
|
for _, env := range envNames {
|
||
|
if value := os.Getenv(env); value != "" {
|
||
|
if err := yaml.Unmarshal([]byte(value), field.Addr().Interface()); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if isBlank := reflect.DeepEqual(field.Interface(), reflect.Zero(field.Type()).Interface()); isBlank {
|
||
|
// Set default configuration if blank
|
||
|
if value := fieldStruct.Tag.Get("default"); value != "" {
|
||
|
if err := yaml.Unmarshal([]byte(value), field.Addr().Interface()); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
} else if fieldStruct.Tag.Get("required") == "true" {
|
||
|
// return error if it is required but blank
|
||
|
return errors.New(fieldStruct.Name + " is required, but blank")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for field.Kind() == reflect.Ptr {
|
||
|
field = field.Elem()
|
||
|
}
|
||
|
|
||
|
if field.Kind() == reflect.Struct {
|
||
|
if err := processTags(field.Addr().Interface(), getPrefixForStruct(prefixes, &fieldStruct)...); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if field.Kind() == reflect.Slice {
|
||
|
for i := 0; i < field.Len(); i++ {
|
||
|
if reflect.Indirect(field.Index(i)).Kind() == reflect.Struct {
|
||
|
if err := processTags(field.Index(i).Addr().Interface(), append(getPrefixForStruct(prefixes, &fieldStruct), fmt.Sprint(i))...); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|