forked from koding/multiconfig
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfile.go
138 lines (117 loc) · 2.79 KB
/
file.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 multiconfig
import (
"encoding/json"
"errors"
"io"
"os"
"path/filepath"
"github.com/BurntSushi/toml"
yaml "gopkg.in/yaml.v3"
)
var (
// ErrSourceNotSet states that neither the path or the reader is set on the loader
ErrSourceNotSet = errors.New("config path or reader is not set")
// ErrFileNotFound states that given file is not exists
ErrFileNotFound = errors.New("config file not found")
)
// TOMLLoader satisifies the loader interface. It loads the configuration from
// the given toml file or Reader.
type TOMLLoader struct {
Path string
Reader io.Reader
}
// Load loads the source into the config defined by struct s
// Defaults to using the Reader if provided, otherwise tries to read from the
// file
func (t *TOMLLoader) Load(s interface{}) error {
var r io.Reader
if t.Reader != nil {
r = t.Reader
} else if t.Path != "" {
file, err := getConfig(t.Path)
if err != nil {
return err
}
defer file.Close()
r = file
} else {
return ErrSourceNotSet
}
if _, err := toml.NewDecoder(r).Decode(s); err != nil {
return err
}
return nil
}
// JSONLoader satisifies the loader interface. It loads the configuration from
// the given json file or Reader.
type JSONLoader struct {
Path string
Reader io.Reader
}
// Load loads the source into the config defined by struct s.
// Defaults to using the Reader if provided, otherwise tries to read from the
// file
func (j *JSONLoader) Load(s interface{}) error {
var r io.Reader
if j.Reader != nil {
r = j.Reader
} else if j.Path != "" {
file, err := getConfig(j.Path)
if err != nil {
return err
}
defer file.Close()
r = file
} else {
return ErrSourceNotSet
}
return json.NewDecoder(r).Decode(s)
}
// YAMLLoader satisifies the loader interface. It loads the configuration from
// the given yaml file.
type YAMLLoader struct {
Path string
Reader io.Reader
}
// Load loads the source into the config defined by struct s.
// Defaults to using the Reader if provided, otherwise tries to read from the
// file
func (y *YAMLLoader) Load(s interface{}) error {
var r io.Reader
if y.Reader != nil {
r = y.Reader
} else if y.Path != "" {
file, err := getConfig(y.Path)
if err != nil {
return err
}
defer file.Close()
r = file
} else {
return ErrSourceNotSet
}
data, err := io.ReadAll(r)
if err != nil {
return err
}
return yaml.Unmarshal(data, s)
}
func getConfig(path string) (*os.File, error) {
pwd, err := os.Getwd()
if err != nil {
return nil, err
}
configPath := path
if !filepath.IsAbs(path) {
configPath = filepath.Join(pwd, path)
}
// check if file with combined path is exists(relative path)
if _, err := os.Stat(configPath); !os.IsNotExist(err) {
return os.Open(configPath)
}
f, err := os.Open(path)
if os.IsNotExist(err) {
return nil, ErrFileNotFound
}
return f, err
}