-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathloader.go
149 lines (121 loc) · 3.62 KB
/
loader.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
package goconf
import (
"fmt"
"os"
"path/filepath"
"github.com/Noah-Huppert/goconf/toml"
"github.com/mitchellh/mapstructure"
"gopkg.in/go-playground/validator.v9"
"gopkg.in/mcuadros/go-defaults.v1"
)
// Loader loads configuration
type Loader struct {
// validate is the validator instance to use for validation
validate *validator.Validate
// formats holds the MapDecoders for file extensions
formats map[string]MapDecoder
// configPaths are paths to files to load
configPaths []string
}
// NewLoader creates a Loader with all MapDecoders implemented by goconf.
func NewLoader() *Loader {
l := &Loader{
validate: validator.New(),
formats: map[string]MapDecoder{},
configPaths: []string{},
}
l.RegisterFormat(".toml", toml.TomlMapDecoder{})
return l
}
// NewDefaultLoader creates a Loader with all MapDecoders implemented by goconf.
func NewDefaultLoader() *Loader {
return NewLoader()
}
// RegisterFormat registers a MapDecoder to be used for a file extension.
// The ext argument should include the final dot and then the extension name.
// An empty string can be passed to target files without an extension.
func (l *Loader) RegisterFormat(ext string, decoder MapDecoder) {
l.formats[ext] = decoder
}
// GetValidate returns the validator instance used to validate configuration
func (l Loader) GetValidate() *validator.Validate {
return l.validate
}
// AddConfigPath adds a potential path from which configuration files will be
// loaded. Must point to file(s) not of directories. The p argument can contain
// shell globs.
func (l *Loader) AddConfigPath(p string) {
// Check if already in configPaths
for _, existingPath := range l.configPaths {
if existingPath == p {
return
}
}
// Add
l.configPaths = append(l.configPaths, p)
}
// Load configuration files into a struct
func (l Loader) Load(c interface{}) error {
// Set default values
defaults.SetDefaults(c)
// Expand config paths
loadPaths := []string{}
for _, configPath := range l.configPaths {
// Interpret shell globs
expandedPaths, err := filepath.Glob(configPath)
if err != nil {
return fmt.Errorf("failed to expand configuration "+
"path \"%s\" glob: %s", configPath,
err.Error())
}
for _, expandedPath := range expandedPaths {
// Check not directory
fi, err := os.Stat(expandedPath)
if err != nil {
return fmt.Errorf("failed to stat "+
"configuration path \"%s\": %s",
expandedPath, err.Error())
}
if fi.IsDir() {
return fmt.Errorf("configuration path "+
"\"%s\" is a directory, cannot be",
expandedPath)
}
// Not directory, add
loadPaths = append(loadPaths, expandedPath)
}
}
// Try to load all files in loadPaths
for _, loadPath := range loadPaths {
// Check if MapDecoder exists for file extension
decoder, ok := l.formats[filepath.Ext(loadPath)]
if !ok {
continue
}
// Use MapDecoder if exists
// Open file
loadFile, err := os.Open(loadPath)
if err != nil {
return fmt.Errorf("error opening configuration "+
"file \"%s\": %s", loadPath, err.Error())
}
// Call MapDecoder
loadMap := map[string]interface{}{}
if err = decoder.Decode(loadFile, &loadMap); err != nil {
return fmt.Errorf("error decoding \"%s\": %s",
loadPath, err.Error())
}
// Put map into struct
if err = mapstructure.Decode(loadMap, c); err != nil {
return fmt.Errorf("error putting decoded map "+
"for \"%s\" into configuration struct: %s",
loadPath, err.Error())
}
}
// Validate configuration struct
if err := l.validate.Struct(c); err != nil {
return fmt.Errorf("failed to validate configuration "+
"struct: %s", err.Error())
}
return nil
}