-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdefaults.go
138 lines (128 loc) · 3.87 KB
/
defaults.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
package cmd_toolkit
import (
"errors"
"fmt"
"github.com/majohn-r/output"
"github.com/spf13/afero"
"gopkg.in/yaml.v3"
"io/fs"
"path/filepath"
)
const (
defaultConfigFileName = "defaults.yaml"
)
var (
defaultConfigurationSettings = map[string]map[string]any{}
)
// AddDefaults copies data from a FlagSet into the map of default configuration settings
func AddDefaults(sf *FlagSet) {
if sf != nil && len(sf.Details) > 0 {
payload := map[string]any{}
for flagName, details := range sf.Details {
bounded, ok := details.DefaultValue.(*IntBounds)
switch ok {
case true:
if bounded != nil {
payload[flagName] = bounded.DefaultValue
}
case false:
payload[flagName] = details.DefaultValue
}
}
defaultConfigurationSettings[sf.Name] = payload
}
}
// WritableDefaults returns the current state of the defaults configuration as a slice of bytes
func WritableDefaults() []byte {
var payload []byte
if len(defaultConfigurationSettings) > 0 {
// ignore error return - we're not dealing in structs, but just maps
payload, _ = yaml.Marshal(defaultConfigurationSettings)
}
return payload
}
// DefaultConfigFileStatus returns the path of the defaults config file and whether that file exists
func DefaultConfigFileStatus() (string, bool) {
path := filepath.Join(ApplicationPath(), defaultConfigFileName)
exists := PlainFileExists(path)
return path, exists
}
// ReadDefaultsConfigFile reads defaults.yaml from the specified path and returns
// a pointer to a cooked Configuration instance; if there is no such file, then
// an empty Configuration is returned and ok is true
func ReadDefaultsConfigFile(o output.Bus) (*Configuration, bool) {
c := EmptyConfiguration()
path := ApplicationPath()
file := filepath.Join(path, defaultConfigFileName)
exists, fileError := verifyDefaultConfigFileExists(o, file)
if fileError != nil {
return c, false
}
if !exists {
return c, true
}
// only probable error circumvented by verifyFileExists failure
rawYaml, _ := afero.ReadFile(fileSystem, file)
data := map[string]any{}
fileError = yaml.Unmarshal(rawYaml, &data)
if fileError != nil {
o.Log(output.Error, "cannot unmarshal yaml content", map[string]any{
"directory": path,
"fileName": defaultConfigFileName,
"error": fileError,
})
o.ErrorPrintf("The configuration file %q is not well-formed YAML: %s.\n", file, ErrorToString(fileError))
o.ErrorPrintln("What to do:")
o.ErrorPrintf("Delete the file %q from %q and restart the application.\n", defaultConfigFileName, path)
return c, false
}
c = newConfiguration(o, data)
o.Log(output.Info, "read configuration file", map[string]any{
"directory": path,
"fileName": defaultConfigFileName,
"value": c,
})
return c, true
}
func reportInvalidConfigurationData(o output.Bus, s string, e error) {
o.ErrorPrintf(
"The configuration file %q contains an invalid value for %q: %s.\n",
defaultConfigFileName,
s,
ErrorToString(e),
)
o.Log(output.Error, "invalid content in configuration file", map[string]any{
"section": s,
"error": e,
})
}
func verifyDefaultConfigFileExists(o output.Bus, path string) (exists bool, err error) {
var f fs.FileInfo
f, err = fileSystem.Stat(path)
switch {
case err == nil:
if f.IsDir() {
o.Log(output.Error, "file is a directory", map[string]any{
"directory": filepath.Dir(path),
"fileName": filepath.Base(path),
})
o.ErrorPrintf("The configuration file %q is a directory.\n", path)
o.ErrorPrintln("What to do:")
o.ErrorPrintf(
"Delete the directory %q from %q and restart the application.\n",
filepath.Base(path),
filepath.Dir(path),
)
err = fmt.Errorf("file exists but is a directory")
} else {
exists = true
}
case errors.Is(err, afero.ErrFileNotFound):
o.Log(output.Info, "file does not exist", map[string]any{
"directory": filepath.Dir(path),
"fileName": filepath.Base(path),
})
err = nil
}
return
}