-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathread.go
124 lines (99 loc) · 2.59 KB
/
read.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
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package conf
import (
"bufio"
"bytes"
"io"
"os"
"strings"
)
// ReadFile reads a file and returns a new configuration representation.
// This representation can be queried with String, etc.
func ReadFile(fname string) (c *Config, err error) {
var file *os.File
if file, err = os.Open(fname); err != nil {
return nil, err
}
c = New()
if err = c.Read(file); err != nil {
return nil, err
}
if err = file.Close(); err != nil {
return nil, err
}
return c, nil
}
func ReadBytes(conf []byte) (c *Config, err error) {
buf := bytes.NewBuffer(conf)
c = New()
if err = c.Read(buf); err != nil {
return nil, err
}
return c, err
}
// Read reads an io.Reader and returns a configuration representation. This
// representation can be queried with String, etc.
func (c *Config) Read(reader io.Reader) (err error) {
buf := bufio.NewReader(reader)
var section, option string
section = "default"
for {
l, buferr := buf.ReadString('\n') // parse line-by-line
l = strings.TrimSpace(l)
if buferr != nil {
if buferr != io.EOF {
return err
}
if len(l) == 0 {
break
}
}
// switch written for readability (not performance)
switch {
case len(l) == 0: // empty line
continue
case l[0] == '#': // comment
continue
case l[0] == ';': // comment
continue
case len(l) >= 3 && strings.ToLower(l[0:3]) == "rem": // comment (for windows users)
continue
case l[0] == '[' && l[len(l)-1] == ']': // new section
option = "" // reset multi-line value
section = strings.TrimSpace(l[1 : len(l)-1])
c.AddSection(section)
case section == "": // not new section and no section defined so far
return ReadError{BlankSection, l}
default: // other alternatives
i := strings.IndexAny(l, "=:")
switch {
case i > 0: // option and value
i := strings.IndexAny(l, "=:")
option = strings.TrimSpace(l[0:i])
value := strings.TrimSpace(stripComments(l[i+1:]))
c.AddOption(section, option, value)
case section != "" && option != "": // continuation of multi-line value
prev, _ := c.RawString(section, option)
value := strings.TrimSpace(stripComments(l))
c.AddOption(section, option, prev+"\n"+value)
default:
return ReadError{CouldNotParse, l}
}
}
// Reached end of file
if buferr == io.EOF {
break
}
}
return nil
}
func stripComments(l string) string {
// comments are preceded by space or TAB
for _, c := range []string{" ;", "\t;", " #", "\t#"} {
if i := strings.Index(l, c); i != -1 {
l = l[0:i]
}
}
return l
}