-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathio.go
93 lines (83 loc) · 1.86 KB
/
io.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
package main
import (
"encoding/json"
"errors"
"fmt"
"io"
"net/url"
"os"
"path/filepath"
yaml "gopkg.in/yaml.v3"
)
func loadURL(u *url.URL) (map[string]interface{}, error) {
if u.Scheme != "file" {
return nil, fmt.Errorf("unsupported %q URL scheme", u.Scheme)
}
return loadFile(filepath.FromSlash(u.Path))
}
func loadFile(pth string) (map[string]interface{}, error) {
f, err := os.Open(pth)
if err != nil {
return nil, err
}
defer f.Close()
var load func(io.Reader) (map[string]interface{}, error)
switch filepath.Ext(pth) {
case ".json":
load = loadJSON
case ".yaml", ".yml":
load = loadYAML
default:
return nil, errors.New("unsupported file extension")
}
return load(f)
}
func loadYAML(r io.Reader) (map[string]interface{}, error) {
data, err := loadAny(yaml.NewDecoder(r))
if err != nil {
return nil, err
}
return fixMaps(data).(map[string]interface{}), err
}
func fixMaps(v interface{}) interface{} {
switch v := v.(type) {
case nil, bool, string, int, int64, float64:
case []interface{}:
for i, item := range v {
v[i] = fixMaps(item)
}
case map[interface{}]interface{}:
m := make(map[string]interface{}, len(v))
for key, val := range v {
m[fmt.Sprint(key)] = fixMaps(val)
}
return m
case map[string]interface{}:
for key, value := range v {
v[key] = fixMaps(value)
}
}
return v
}
func loadJSON(r io.Reader) (map[string]interface{}, error) {
dec := json.NewDecoder(r)
data, err := loadAny(dec)
if err != nil {
return nil, err
}
if dec.More() {
return nil, errors.New("unexpected data after JSON content")
}
return data, err
}
func loadAny(decoder interface{ Decode(interface{}) error }) (map[string]interface{}, error) {
var data map[string]interface{}
err := decoder.Decode(&data)
if err != nil {
return nil, err
}
if data == nil {
return nil, errors.New("unexpected empty object")
}
return data, err
}