mirror of
https://github.com/thegeeklab/drone-s3-sync.git
synced 2024-11-25 20:30:42 +00:00
1227 lines
29 KiB
Go
1227 lines
29 KiB
Go
|
// Copyright 2014 Unknwon
|
||
|
//
|
||
|
// Licensed under the Apache License, Version 2.0 (the "License"): you may
|
||
|
// not use this file except in compliance with the License. You may obtain
|
||
|
// a copy of the License at
|
||
|
//
|
||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||
|
//
|
||
|
// Unless required by applicable law or agreed to in writing, software
|
||
|
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||
|
// License for the specific language governing permissions and limitations
|
||
|
// under the License.
|
||
|
|
||
|
// Package ini provides INI file read and write functionality in Go.
|
||
|
package ini
|
||
|
|
||
|
import (
|
||
|
"bufio"
|
||
|
"bytes"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"os"
|
||
|
"regexp"
|
||
|
"runtime"
|
||
|
"strconv"
|
||
|
"strings"
|
||
|
"sync"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
DEFAULT_SECTION = "DEFAULT"
|
||
|
// Maximum allowed depth when recursively substituing variable names.
|
||
|
_DEPTH_VALUES = 99
|
||
|
|
||
|
_VERSION = "1.6.0"
|
||
|
)
|
||
|
|
||
|
func Version() string {
|
||
|
return _VERSION
|
||
|
}
|
||
|
|
||
|
var (
|
||
|
LineBreak = "\n"
|
||
|
|
||
|
// Variable regexp pattern: %(variable)s
|
||
|
varPattern = regexp.MustCompile(`%\(([^\)]+)\)s`)
|
||
|
|
||
|
// Write spaces around "=" to look better.
|
||
|
PrettyFormat = true
|
||
|
)
|
||
|
|
||
|
func init() {
|
||
|
if runtime.GOOS == "windows" {
|
||
|
LineBreak = "\r\n"
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func inSlice(str string, s []string) bool {
|
||
|
for _, v := range s {
|
||
|
if str == v {
|
||
|
return true
|
||
|
}
|
||
|
}
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
// dataSource is a interface that returns file content.
|
||
|
type dataSource interface {
|
||
|
ReadCloser() (io.ReadCloser, error)
|
||
|
}
|
||
|
|
||
|
type sourceFile struct {
|
||
|
name string
|
||
|
}
|
||
|
|
||
|
func (s sourceFile) ReadCloser() (_ io.ReadCloser, err error) {
|
||
|
return os.Open(s.name)
|
||
|
}
|
||
|
|
||
|
type bytesReadCloser struct {
|
||
|
reader io.Reader
|
||
|
}
|
||
|
|
||
|
func (rc *bytesReadCloser) Read(p []byte) (n int, err error) {
|
||
|
return rc.reader.Read(p)
|
||
|
}
|
||
|
|
||
|
func (rc *bytesReadCloser) Close() error {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
type sourceData struct {
|
||
|
data []byte
|
||
|
}
|
||
|
|
||
|
func (s *sourceData) ReadCloser() (io.ReadCloser, error) {
|
||
|
return &bytesReadCloser{bytes.NewReader(s.data)}, nil
|
||
|
}
|
||
|
|
||
|
// ____ __.
|
||
|
// | |/ _|____ ___.__.
|
||
|
// | <_/ __ < | |
|
||
|
// | | \ ___/\___ |
|
||
|
// |____|__ \___ > ____|
|
||
|
// \/ \/\/
|
||
|
|
||
|
// Key represents a key under a section.
|
||
|
type Key struct {
|
||
|
s *Section
|
||
|
Comment string
|
||
|
name string
|
||
|
value string
|
||
|
isAutoIncr bool
|
||
|
}
|
||
|
|
||
|
// Name returns name of key.
|
||
|
func (k *Key) Name() string {
|
||
|
return k.name
|
||
|
}
|
||
|
|
||
|
// Value returns raw value of key for performance purpose.
|
||
|
func (k *Key) Value() string {
|
||
|
return k.value
|
||
|
}
|
||
|
|
||
|
// String returns string representation of value.
|
||
|
func (k *Key) String() string {
|
||
|
val := k.value
|
||
|
if strings.Index(val, "%") == -1 {
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
for i := 0; i < _DEPTH_VALUES; i++ {
|
||
|
vr := varPattern.FindString(val)
|
||
|
if len(vr) == 0 {
|
||
|
break
|
||
|
}
|
||
|
|
||
|
// Take off leading '%(' and trailing ')s'.
|
||
|
noption := strings.TrimLeft(vr, "%(")
|
||
|
noption = strings.TrimRight(noption, ")s")
|
||
|
|
||
|
// Search in the same section.
|
||
|
nk, err := k.s.GetKey(noption)
|
||
|
if err != nil {
|
||
|
// Search again in default section.
|
||
|
nk, _ = k.s.f.Section("").GetKey(noption)
|
||
|
}
|
||
|
|
||
|
// Substitute by new value and take off leading '%(' and trailing ')s'.
|
||
|
val = strings.Replace(val, vr, nk.value, -1)
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// Validate accepts a validate function which can
|
||
|
// return modifed result as key value.
|
||
|
func (k *Key) Validate(fn func(string) string) string {
|
||
|
return fn(k.String())
|
||
|
}
|
||
|
|
||
|
// parseBool returns the boolean value represented by the string.
|
||
|
//
|
||
|
// It accepts 1, t, T, TRUE, true, True, YES, yes, Yes, ON, on, On,
|
||
|
// 0, f, F, FALSE, false, False, NO, no, No, OFF, off, Off.
|
||
|
// Any other value returns an error.
|
||
|
func parseBool(str string) (value bool, err error) {
|
||
|
switch str {
|
||
|
case "1", "t", "T", "true", "TRUE", "True", "YES", "yes", "Yes", "ON", "on", "On":
|
||
|
return true, nil
|
||
|
case "0", "f", "F", "false", "FALSE", "False", "NO", "no", "No", "OFF", "off", "Off":
|
||
|
return false, nil
|
||
|
}
|
||
|
return false, fmt.Errorf("parsing \"%s\": invalid syntax", str)
|
||
|
}
|
||
|
|
||
|
// Bool returns bool type value.
|
||
|
func (k *Key) Bool() (bool, error) {
|
||
|
return parseBool(k.String())
|
||
|
}
|
||
|
|
||
|
// Float64 returns float64 type value.
|
||
|
func (k *Key) Float64() (float64, error) {
|
||
|
return strconv.ParseFloat(k.String(), 64)
|
||
|
}
|
||
|
|
||
|
// Int returns int type value.
|
||
|
func (k *Key) Int() (int, error) {
|
||
|
return strconv.Atoi(k.String())
|
||
|
}
|
||
|
|
||
|
// Int64 returns int64 type value.
|
||
|
func (k *Key) Int64() (int64, error) {
|
||
|
return strconv.ParseInt(k.String(), 10, 64)
|
||
|
}
|
||
|
|
||
|
// Uint returns uint type valued.
|
||
|
func (k *Key) Uint() (uint, error) {
|
||
|
u, e := strconv.ParseUint(k.String(), 10, 64)
|
||
|
return uint(u), e
|
||
|
}
|
||
|
|
||
|
// Uint64 returns uint64 type value.
|
||
|
func (k *Key) Uint64() (uint64, error) {
|
||
|
return strconv.ParseUint(k.String(), 10, 64)
|
||
|
}
|
||
|
|
||
|
// Duration returns time.Duration type value.
|
||
|
func (k *Key) Duration() (time.Duration, error) {
|
||
|
return time.ParseDuration(k.String())
|
||
|
}
|
||
|
|
||
|
// TimeFormat parses with given format and returns time.Time type value.
|
||
|
func (k *Key) TimeFormat(format string) (time.Time, error) {
|
||
|
return time.Parse(format, k.String())
|
||
|
}
|
||
|
|
||
|
// Time parses with RFC3339 format and returns time.Time type value.
|
||
|
func (k *Key) Time() (time.Time, error) {
|
||
|
return k.TimeFormat(time.RFC3339)
|
||
|
}
|
||
|
|
||
|
// MustString returns default value if key value is empty.
|
||
|
func (k *Key) MustString(defaultVal string) string {
|
||
|
val := k.String()
|
||
|
if len(val) == 0 {
|
||
|
return defaultVal
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// MustBool always returns value without error,
|
||
|
// it returns false if error occurs.
|
||
|
func (k *Key) MustBool(defaultVal ...bool) bool {
|
||
|
val, err := k.Bool()
|
||
|
if len(defaultVal) > 0 && err != nil {
|
||
|
return defaultVal[0]
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// MustFloat64 always returns value without error,
|
||
|
// it returns 0.0 if error occurs.
|
||
|
func (k *Key) MustFloat64(defaultVal ...float64) float64 {
|
||
|
val, err := k.Float64()
|
||
|
if len(defaultVal) > 0 && err != nil {
|
||
|
return defaultVal[0]
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// MustInt always returns value without error,
|
||
|
// it returns 0 if error occurs.
|
||
|
func (k *Key) MustInt(defaultVal ...int) int {
|
||
|
val, err := k.Int()
|
||
|
if len(defaultVal) > 0 && err != nil {
|
||
|
return defaultVal[0]
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// MustInt64 always returns value without error,
|
||
|
// it returns 0 if error occurs.
|
||
|
func (k *Key) MustInt64(defaultVal ...int64) int64 {
|
||
|
val, err := k.Int64()
|
||
|
if len(defaultVal) > 0 && err != nil {
|
||
|
return defaultVal[0]
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// MustUint always returns value without error,
|
||
|
// it returns 0 if error occurs.
|
||
|
func (k *Key) MustUint(defaultVal ...uint) uint {
|
||
|
val, err := k.Uint()
|
||
|
if len(defaultVal) > 0 && err != nil {
|
||
|
return defaultVal[0]
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// MustUint64 always returns value without error,
|
||
|
// it returns 0 if error occurs.
|
||
|
func (k *Key) MustUint64(defaultVal ...uint64) uint64 {
|
||
|
val, err := k.Uint64()
|
||
|
if len(defaultVal) > 0 && err != nil {
|
||
|
return defaultVal[0]
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// MustDuration always returns value without error,
|
||
|
// it returns zero value if error occurs.
|
||
|
func (k *Key) MustDuration(defaultVal ...time.Duration) time.Duration {
|
||
|
val, err := k.Duration()
|
||
|
if len(defaultVal) > 0 && err != nil {
|
||
|
return defaultVal[0]
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// MustTimeFormat always parses with given format and returns value without error,
|
||
|
// it returns zero value if error occurs.
|
||
|
func (k *Key) MustTimeFormat(format string, defaultVal ...time.Time) time.Time {
|
||
|
val, err := k.TimeFormat(format)
|
||
|
if len(defaultVal) > 0 && err != nil {
|
||
|
return defaultVal[0]
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// MustTime always parses with RFC3339 format and returns value without error,
|
||
|
// it returns zero value if error occurs.
|
||
|
func (k *Key) MustTime(defaultVal ...time.Time) time.Time {
|
||
|
return k.MustTimeFormat(time.RFC3339, defaultVal...)
|
||
|
}
|
||
|
|
||
|
// In always returns value without error,
|
||
|
// it returns default value if error occurs or doesn't fit into candidates.
|
||
|
func (k *Key) In(defaultVal string, candidates []string) string {
|
||
|
val := k.String()
|
||
|
for _, cand := range candidates {
|
||
|
if val == cand {
|
||
|
return val
|
||
|
}
|
||
|
}
|
||
|
return defaultVal
|
||
|
}
|
||
|
|
||
|
// InFloat64 always returns value without error,
|
||
|
// it returns default value if error occurs or doesn't fit into candidates.
|
||
|
func (k *Key) InFloat64(defaultVal float64, candidates []float64) float64 {
|
||
|
val := k.MustFloat64()
|
||
|
for _, cand := range candidates {
|
||
|
if val == cand {
|
||
|
return val
|
||
|
}
|
||
|
}
|
||
|
return defaultVal
|
||
|
}
|
||
|
|
||
|
// InInt always returns value without error,
|
||
|
// it returns default value if error occurs or doesn't fit into candidates.
|
||
|
func (k *Key) InInt(defaultVal int, candidates []int) int {
|
||
|
val := k.MustInt()
|
||
|
for _, cand := range candidates {
|
||
|
if val == cand {
|
||
|
return val
|
||
|
}
|
||
|
}
|
||
|
return defaultVal
|
||
|
}
|
||
|
|
||
|
// InInt64 always returns value without error,
|
||
|
// it returns default value if error occurs or doesn't fit into candidates.
|
||
|
func (k *Key) InInt64(defaultVal int64, candidates []int64) int64 {
|
||
|
val := k.MustInt64()
|
||
|
for _, cand := range candidates {
|
||
|
if val == cand {
|
||
|
return val
|
||
|
}
|
||
|
}
|
||
|
return defaultVal
|
||
|
}
|
||
|
|
||
|
// InUint always returns value without error,
|
||
|
// it returns default value if error occurs or doesn't fit into candidates.
|
||
|
func (k *Key) InUint(defaultVal uint, candidates []uint) uint {
|
||
|
val := k.MustUint()
|
||
|
for _, cand := range candidates {
|
||
|
if val == cand {
|
||
|
return val
|
||
|
}
|
||
|
}
|
||
|
return defaultVal
|
||
|
}
|
||
|
|
||
|
// InUint64 always returns value without error,
|
||
|
// it returns default value if error occurs or doesn't fit into candidates.
|
||
|
func (k *Key) InUint64(defaultVal uint64, candidates []uint64) uint64 {
|
||
|
val := k.MustUint64()
|
||
|
for _, cand := range candidates {
|
||
|
if val == cand {
|
||
|
return val
|
||
|
}
|
||
|
}
|
||
|
return defaultVal
|
||
|
}
|
||
|
|
||
|
// InTimeFormat always parses with given format and returns value without error,
|
||
|
// it returns default value if error occurs or doesn't fit into candidates.
|
||
|
func (k *Key) InTimeFormat(format string, defaultVal time.Time, candidates []time.Time) time.Time {
|
||
|
val := k.MustTimeFormat(format)
|
||
|
for _, cand := range candidates {
|
||
|
if val == cand {
|
||
|
return val
|
||
|
}
|
||
|
}
|
||
|
return defaultVal
|
||
|
}
|
||
|
|
||
|
// InTime always parses with RFC3339 format and returns value without error,
|
||
|
// it returns default value if error occurs or doesn't fit into candidates.
|
||
|
func (k *Key) InTime(defaultVal time.Time, candidates []time.Time) time.Time {
|
||
|
return k.InTimeFormat(time.RFC3339, defaultVal, candidates)
|
||
|
}
|
||
|
|
||
|
// RangeFloat64 checks if value is in given range inclusively,
|
||
|
// and returns default value if it's not.
|
||
|
func (k *Key) RangeFloat64(defaultVal, min, max float64) float64 {
|
||
|
val := k.MustFloat64()
|
||
|
if val < min || val > max {
|
||
|
return defaultVal
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// RangeInt checks if value is in given range inclusively,
|
||
|
// and returns default value if it's not.
|
||
|
func (k *Key) RangeInt(defaultVal, min, max int) int {
|
||
|
val := k.MustInt()
|
||
|
if val < min || val > max {
|
||
|
return defaultVal
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// RangeInt64 checks if value is in given range inclusively,
|
||
|
// and returns default value if it's not.
|
||
|
func (k *Key) RangeInt64(defaultVal, min, max int64) int64 {
|
||
|
val := k.MustInt64()
|
||
|
if val < min || val > max {
|
||
|
return defaultVal
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// RangeTimeFormat checks if value with given format is in given range inclusively,
|
||
|
// and returns default value if it's not.
|
||
|
func (k *Key) RangeTimeFormat(format string, defaultVal, min, max time.Time) time.Time {
|
||
|
val := k.MustTimeFormat(format)
|
||
|
if val.Unix() < min.Unix() || val.Unix() > max.Unix() {
|
||
|
return defaultVal
|
||
|
}
|
||
|
return val
|
||
|
}
|
||
|
|
||
|
// RangeTime checks if value with RFC3339 format is in given range inclusively,
|
||
|
// and returns default value if it's not.
|
||
|
func (k *Key) RangeTime(defaultVal, min, max time.Time) time.Time {
|
||
|
return k.RangeTimeFormat(time.RFC3339, defaultVal, min, max)
|
||
|
}
|
||
|
|
||
|
// Strings returns list of string devide by given delimiter.
|
||
|
func (k *Key) Strings(delim string) []string {
|
||
|
str := k.String()
|
||
|
if len(str) == 0 {
|
||
|
return []string{}
|
||
|
}
|
||
|
|
||
|
vals := strings.Split(str, delim)
|
||
|
for i := range vals {
|
||
|
vals[i] = strings.TrimSpace(vals[i])
|
||
|
}
|
||
|
return vals
|
||
|
}
|
||
|
|
||
|
// Float64s returns list of float64 devide by given delimiter.
|
||
|
func (k *Key) Float64s(delim string) []float64 {
|
||
|
strs := k.Strings(delim)
|
||
|
vals := make([]float64, len(strs))
|
||
|
for i := range strs {
|
||
|
vals[i], _ = strconv.ParseFloat(strs[i], 64)
|
||
|
}
|
||
|
return vals
|
||
|
}
|
||
|
|
||
|
// Ints returns list of int devide by given delimiter.
|
||
|
func (k *Key) Ints(delim string) []int {
|
||
|
strs := k.Strings(delim)
|
||
|
vals := make([]int, len(strs))
|
||
|
for i := range strs {
|
||
|
vals[i], _ = strconv.Atoi(strs[i])
|
||
|
}
|
||
|
return vals
|
||
|
}
|
||
|
|
||
|
// Int64s returns list of int64 devide by given delimiter.
|
||
|
func (k *Key) Int64s(delim string) []int64 {
|
||
|
strs := k.Strings(delim)
|
||
|
vals := make([]int64, len(strs))
|
||
|
for i := range strs {
|
||
|
vals[i], _ = strconv.ParseInt(strs[i], 10, 64)
|
||
|
}
|
||
|
return vals
|
||
|
}
|
||
|
|
||
|
// Uints returns list of uint devide by given delimiter.
|
||
|
func (k *Key) Uints(delim string) []uint {
|
||
|
strs := k.Strings(delim)
|
||
|
vals := make([]uint, len(strs))
|
||
|
for i := range strs {
|
||
|
u, _ := strconv.ParseUint(strs[i], 10, 64)
|
||
|
vals[i] = uint(u)
|
||
|
}
|
||
|
return vals
|
||
|
}
|
||
|
|
||
|
// Uint64s returns list of uint64 devide by given delimiter.
|
||
|
func (k *Key) Uint64s(delim string) []uint64 {
|
||
|
strs := k.Strings(delim)
|
||
|
vals := make([]uint64, len(strs))
|
||
|
for i := range strs {
|
||
|
vals[i], _ = strconv.ParseUint(strs[i], 10, 64)
|
||
|
}
|
||
|
return vals
|
||
|
}
|
||
|
|
||
|
// TimesFormat parses with given format and returns list of time.Time devide by given delimiter.
|
||
|
func (k *Key) TimesFormat(format, delim string) []time.Time {
|
||
|
strs := k.Strings(delim)
|
||
|
vals := make([]time.Time, len(strs))
|
||
|
for i := range strs {
|
||
|
vals[i], _ = time.Parse(format, strs[i])
|
||
|
}
|
||
|
return vals
|
||
|
}
|
||
|
|
||
|
// Times parses with RFC3339 format and returns list of time.Time devide by given delimiter.
|
||
|
func (k *Key) Times(delim string) []time.Time {
|
||
|
return k.TimesFormat(time.RFC3339, delim)
|
||
|
}
|
||
|
|
||
|
// SetValue changes key value.
|
||
|
func (k *Key) SetValue(v string) {
|
||
|
k.value = v
|
||
|
}
|
||
|
|
||
|
// _________ __ .__
|
||
|
// / _____/ ____ _____/ |_|__| ____ ____
|
||
|
// \_____ \_/ __ \_/ ___\ __\ |/ _ \ / \
|
||
|
// / \ ___/\ \___| | | ( <_> ) | \
|
||
|
// /_______ /\___ >\___ >__| |__|\____/|___| /
|
||
|
// \/ \/ \/ \/
|
||
|
|
||
|
// Section represents a config section.
|
||
|
type Section struct {
|
||
|
f *File
|
||
|
Comment string
|
||
|
name string
|
||
|
keys map[string]*Key
|
||
|
keyList []string
|
||
|
keysHash map[string]string
|
||
|
}
|
||
|
|
||
|
func newSection(f *File, name string) *Section {
|
||
|
return &Section{f, "", name, make(map[string]*Key), make([]string, 0, 10), make(map[string]string)}
|
||
|
}
|
||
|
|
||
|
// Name returns name of Section.
|
||
|
func (s *Section) Name() string {
|
||
|
return s.name
|
||
|
}
|
||
|
|
||
|
// NewKey creates a new key to given section.
|
||
|
func (s *Section) NewKey(name, val string) (*Key, error) {
|
||
|
if len(name) == 0 {
|
||
|
return nil, errors.New("error creating new key: empty key name")
|
||
|
}
|
||
|
|
||
|
if s.f.BlockMode {
|
||
|
s.f.lock.Lock()
|
||
|
defer s.f.lock.Unlock()
|
||
|
}
|
||
|
|
||
|
if inSlice(name, s.keyList) {
|
||
|
s.keys[name].value = val
|
||
|
return s.keys[name], nil
|
||
|
}
|
||
|
|
||
|
s.keyList = append(s.keyList, name)
|
||
|
s.keys[name] = &Key{s, "", name, val, false}
|
||
|
s.keysHash[name] = val
|
||
|
return s.keys[name], nil
|
||
|
}
|
||
|
|
||
|
// GetKey returns key in section by given name.
|
||
|
func (s *Section) GetKey(name string) (*Key, error) {
|
||
|
// FIXME: change to section level lock?
|
||
|
if s.f.BlockMode {
|
||
|
s.f.lock.RLock()
|
||
|
}
|
||
|
key := s.keys[name]
|
||
|
if s.f.BlockMode {
|
||
|
s.f.lock.RUnlock()
|
||
|
}
|
||
|
|
||
|
if key == nil {
|
||
|
// Check if it is a child-section.
|
||
|
sname := s.name
|
||
|
for {
|
||
|
if i := strings.LastIndex(sname, "."); i > -1 {
|
||
|
sname = sname[:i]
|
||
|
sec, err := s.f.GetSection(sname)
|
||
|
if err != nil {
|
||
|
continue
|
||
|
}
|
||
|
return sec.GetKey(name)
|
||
|
} else {
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
return nil, fmt.Errorf("error when getting key of section '%s': key '%s' not exists", s.name, name)
|
||
|
}
|
||
|
return key, nil
|
||
|
}
|
||
|
|
||
|
// Key assumes named Key exists in section and returns a zero-value when not.
|
||
|
func (s *Section) Key(name string) *Key {
|
||
|
key, err := s.GetKey(name)
|
||
|
if err != nil {
|
||
|
// It's OK here because the only possible error is empty key name,
|
||
|
// but if it's empty, this piece of code won't be executed.
|
||
|
key, _ = s.NewKey(name, "")
|
||
|
return key
|
||
|
}
|
||
|
return key
|
||
|
}
|
||
|
|
||
|
// Keys returns list of keys of section.
|
||
|
func (s *Section) Keys() []*Key {
|
||
|
keys := make([]*Key, len(s.keyList))
|
||
|
for i := range s.keyList {
|
||
|
keys[i] = s.Key(s.keyList[i])
|
||
|
}
|
||
|
return keys
|
||
|
}
|
||
|
|
||
|
// KeyStrings returns list of key names of section.
|
||
|
func (s *Section) KeyStrings() []string {
|
||
|
list := make([]string, len(s.keyList))
|
||
|
copy(list, s.keyList)
|
||
|
return list
|
||
|
}
|
||
|
|
||
|
// KeysHash returns keys hash consisting of names and values.
|
||
|
func (s *Section) KeysHash() map[string]string {
|
||
|
if s.f.BlockMode {
|
||
|
s.f.lock.RLock()
|
||
|
defer s.f.lock.RUnlock()
|
||
|
}
|
||
|
|
||
|
hash := map[string]string{}
|
||
|
for key, value := range s.keysHash {
|
||
|
hash[key] = value
|
||
|
}
|
||
|
return hash
|
||
|
}
|
||
|
|
||
|
// DeleteKey deletes a key from section.
|
||
|
func (s *Section) DeleteKey(name string) {
|
||
|
if s.f.BlockMode {
|
||
|
s.f.lock.Lock()
|
||
|
defer s.f.lock.Unlock()
|
||
|
}
|
||
|
|
||
|
for i, k := range s.keyList {
|
||
|
if k == name {
|
||
|
s.keyList = append(s.keyList[:i], s.keyList[i+1:]...)
|
||
|
delete(s.keys, name)
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// ___________.__.__
|
||
|
// \_ _____/|__| | ____
|
||
|
// | __) | | | _/ __ \
|
||
|
// | \ | | |_\ ___/
|
||
|
// \___ / |__|____/\___ >
|
||
|
// \/ \/
|
||
|
|
||
|
// File represents a combination of a or more INI file(s) in memory.
|
||
|
type File struct {
|
||
|
// Should make things safe, but sometimes doesn't matter.
|
||
|
BlockMode bool
|
||
|
// Make sure data is safe in multiple goroutines.
|
||
|
lock sync.RWMutex
|
||
|
|
||
|
// Allow combination of multiple data sources.
|
||
|
dataSources []dataSource
|
||
|
// Actual data is stored here.
|
||
|
sections map[string]*Section
|
||
|
|
||
|
// To keep data in order.
|
||
|
sectionList []string
|
||
|
|
||
|
NameMapper
|
||
|
}
|
||
|
|
||
|
// newFile initializes File object with given data sources.
|
||
|
func newFile(dataSources []dataSource) *File {
|
||
|
return &File{
|
||
|
BlockMode: true,
|
||
|
dataSources: dataSources,
|
||
|
sections: make(map[string]*Section),
|
||
|
sectionList: make([]string, 0, 10),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func parseDataSource(source interface{}) (dataSource, error) {
|
||
|
switch s := source.(type) {
|
||
|
case string:
|
||
|
return sourceFile{s}, nil
|
||
|
case []byte:
|
||
|
return &sourceData{s}, nil
|
||
|
default:
|
||
|
return nil, fmt.Errorf("error parsing data source: unknown type '%s'", s)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Load loads and parses from INI data sources.
|
||
|
// Arguments can be mixed of file name with string type, or raw data in []byte.
|
||
|
func Load(source interface{}, others ...interface{}) (_ *File, err error) {
|
||
|
sources := make([]dataSource, len(others)+1)
|
||
|
sources[0], err = parseDataSource(source)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
for i := range others {
|
||
|
sources[i+1], err = parseDataSource(others[i])
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
}
|
||
|
f := newFile(sources)
|
||
|
return f, f.Reload()
|
||
|
}
|
||
|
|
||
|
// Empty returns an empty file object.
|
||
|
func Empty() *File {
|
||
|
// Ignore error here, we sure our data is good.
|
||
|
f, _ := Load([]byte(""))
|
||
|
return f
|
||
|
}
|
||
|
|
||
|
// NewSection creates a new section.
|
||
|
func (f *File) NewSection(name string) (*Section, error) {
|
||
|
if len(name) == 0 {
|
||
|
return nil, errors.New("error creating new section: empty section name")
|
||
|
}
|
||
|
|
||
|
if f.BlockMode {
|
||
|
f.lock.Lock()
|
||
|
defer f.lock.Unlock()
|
||
|
}
|
||
|
|
||
|
if inSlice(name, f.sectionList) {
|
||
|
return f.sections[name], nil
|
||
|
}
|
||
|
|
||
|
f.sectionList = append(f.sectionList, name)
|
||
|
f.sections[name] = newSection(f, name)
|
||
|
return f.sections[name], nil
|
||
|
}
|
||
|
|
||
|
// NewSections creates a list of sections.
|
||
|
func (f *File) NewSections(names ...string) (err error) {
|
||
|
for _, name := range names {
|
||
|
if _, err = f.NewSection(name); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// GetSection returns section by given name.
|
||
|
func (f *File) GetSection(name string) (*Section, error) {
|
||
|
if len(name) == 0 {
|
||
|
name = DEFAULT_SECTION
|
||
|
}
|
||
|
|
||
|
if f.BlockMode {
|
||
|
f.lock.RLock()
|
||
|
defer f.lock.RUnlock()
|
||
|
}
|
||
|
|
||
|
sec := f.sections[name]
|
||
|
if sec == nil {
|
||
|
return nil, fmt.Errorf("error when getting section: section '%s' not exists", name)
|
||
|
}
|
||
|
return sec, nil
|
||
|
}
|
||
|
|
||
|
// Section assumes named section exists and returns a zero-value when not.
|
||
|
func (f *File) Section(name string) *Section {
|
||
|
sec, err := f.GetSection(name)
|
||
|
if err != nil {
|
||
|
// Note: It's OK here because the only possible error is empty section name,
|
||
|
// but if it's empty, this piece of code won't be executed.
|
||
|
sec, _ = f.NewSection(name)
|
||
|
return sec
|
||
|
}
|
||
|
return sec
|
||
|
}
|
||
|
|
||
|
// Section returns list of Section.
|
||
|
func (f *File) Sections() []*Section {
|
||
|
sections := make([]*Section, len(f.sectionList))
|
||
|
for i := range f.sectionList {
|
||
|
sections[i] = f.Section(f.sectionList[i])
|
||
|
}
|
||
|
return sections
|
||
|
}
|
||
|
|
||
|
// SectionStrings returns list of section names.
|
||
|
func (f *File) SectionStrings() []string {
|
||
|
list := make([]string, len(f.sectionList))
|
||
|
copy(list, f.sectionList)
|
||
|
return list
|
||
|
}
|
||
|
|
||
|
// DeleteSection deletes a section.
|
||
|
func (f *File) DeleteSection(name string) {
|
||
|
if f.BlockMode {
|
||
|
f.lock.Lock()
|
||
|
defer f.lock.Unlock()
|
||
|
}
|
||
|
|
||
|
if len(name) == 0 {
|
||
|
name = DEFAULT_SECTION
|
||
|
}
|
||
|
|
||
|
for i, s := range f.sectionList {
|
||
|
if s == name {
|
||
|
f.sectionList = append(f.sectionList[:i], f.sectionList[i+1:]...)
|
||
|
delete(f.sections, name)
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func cutComment(str string) string {
|
||
|
i := strings.Index(str, "#")
|
||
|
if i == -1 {
|
||
|
return str
|
||
|
}
|
||
|
return str[:i]
|
||
|
}
|
||
|
|
||
|
func checkMultipleLines(buf *bufio.Reader, line, val, valQuote string) (string, error) {
|
||
|
isEnd := false
|
||
|
for {
|
||
|
next, err := buf.ReadString('\n')
|
||
|
if err != nil {
|
||
|
if err != io.EOF {
|
||
|
return "", err
|
||
|
}
|
||
|
isEnd = true
|
||
|
}
|
||
|
pos := strings.LastIndex(next, valQuote)
|
||
|
if pos > -1 {
|
||
|
val += next[:pos]
|
||
|
break
|
||
|
}
|
||
|
val += next
|
||
|
if isEnd {
|
||
|
return "", fmt.Errorf("error parsing line: missing closing key quote from '%s' to '%s'", line, next)
|
||
|
}
|
||
|
}
|
||
|
return val, nil
|
||
|
}
|
||
|
|
||
|
func checkContinuationLines(buf *bufio.Reader, val string) (string, bool, error) {
|
||
|
isEnd := false
|
||
|
for {
|
||
|
valLen := len(val)
|
||
|
if valLen == 0 || val[valLen-1] != '\\' {
|
||
|
break
|
||
|
}
|
||
|
val = val[:valLen-1]
|
||
|
|
||
|
next, err := buf.ReadString('\n')
|
||
|
if err != nil {
|
||
|
if err != io.EOF {
|
||
|
return "", isEnd, err
|
||
|
}
|
||
|
isEnd = true
|
||
|
}
|
||
|
|
||
|
next = strings.TrimSpace(next)
|
||
|
if len(next) == 0 {
|
||
|
break
|
||
|
}
|
||
|
val += next
|
||
|
}
|
||
|
return val, isEnd, nil
|
||
|
}
|
||
|
|
||
|
// parse parses data through an io.Reader.
|
||
|
func (f *File) parse(reader io.Reader) error {
|
||
|
buf := bufio.NewReader(reader)
|
||
|
|
||
|
// Handle BOM-UTF8.
|
||
|
// http://en.wikipedia.org/wiki/Byte_order_mark#Representations_of_byte_order_marks_by_encoding
|
||
|
mask, err := buf.Peek(3)
|
||
|
if err == nil && len(mask) >= 3 && mask[0] == 239 && mask[1] == 187 && mask[2] == 191 {
|
||
|
buf.Read(mask)
|
||
|
}
|
||
|
|
||
|
count := 1
|
||
|
comments := ""
|
||
|
isEnd := false
|
||
|
|
||
|
section, err := f.NewSection(DEFAULT_SECTION)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
for {
|
||
|
line, err := buf.ReadString('\n')
|
||
|
line = strings.TrimSpace(line)
|
||
|
length := len(line)
|
||
|
|
||
|
// Check error and ignore io.EOF just for a moment.
|
||
|
if err != nil {
|
||
|
if err != io.EOF {
|
||
|
return fmt.Errorf("error reading next line: %v", err)
|
||
|
}
|
||
|
// The last line of file could be an empty line.
|
||
|
if length == 0 {
|
||
|
break
|
||
|
}
|
||
|
isEnd = true
|
||
|
}
|
||
|
|
||
|
// Skip empty lines.
|
||
|
if length == 0 {
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
switch {
|
||
|
case line[0] == '#' || line[0] == ';': // Comments.
|
||
|
if len(comments) == 0 {
|
||
|
comments = line
|
||
|
} else {
|
||
|
comments += LineBreak + line
|
||
|
}
|
||
|
continue
|
||
|
case line[0] == '[' && line[length-1] == ']': // New sction.
|
||
|
section, err = f.NewSection(strings.TrimSpace(line[1 : length-1]))
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
if len(comments) > 0 {
|
||
|
section.Comment = comments
|
||
|
comments = ""
|
||
|
}
|
||
|
// Reset counter.
|
||
|
count = 1
|
||
|
continue
|
||
|
}
|
||
|
|
||
|
// Other possibilities.
|
||
|
var (
|
||
|
i int
|
||
|
keyQuote string
|
||
|
kname string
|
||
|
valQuote string
|
||
|
val string
|
||
|
)
|
||
|
|
||
|
// Key name surrounded by quotes.
|
||
|
if line[0] == '"' {
|
||
|
if length > 6 && line[0:3] == `"""` {
|
||
|
keyQuote = `"""`
|
||
|
} else {
|
||
|
keyQuote = `"`
|
||
|
}
|
||
|
} else if line[0] == '`' {
|
||
|
keyQuote = "`"
|
||
|
}
|
||
|
if len(keyQuote) > 0 {
|
||
|
qLen := len(keyQuote)
|
||
|
pos := strings.Index(line[qLen:], keyQuote)
|
||
|
if pos == -1 {
|
||
|
return fmt.Errorf("error parsing line: missing closing key quote: %s", line)
|
||
|
}
|
||
|
pos = pos + qLen
|
||
|
i = strings.IndexAny(line[pos:], "=:")
|
||
|
if i < 0 {
|
||
|
return fmt.Errorf("error parsing line: key-value delimiter not found: %s", line)
|
||
|
} else if i == pos {
|
||
|
return fmt.Errorf("error parsing line: key is empty: %s", line)
|
||
|
}
|
||
|
i = i + pos
|
||
|
kname = line[qLen:pos] // Just keep spaces inside quotes.
|
||
|
} else {
|
||
|
i = strings.IndexAny(line, "=:")
|
||
|
if i < 0 {
|
||
|
return fmt.Errorf("error parsing line: key-value delimiter not found: %s", line)
|
||
|
} else if i == 0 {
|
||
|
return fmt.Errorf("error parsing line: key is empty: %s", line)
|
||
|
}
|
||
|
kname = strings.TrimSpace(line[0:i])
|
||
|
}
|
||
|
|
||
|
isAutoIncr := false
|
||
|
// Auto increment.
|
||
|
if kname == "-" {
|
||
|
isAutoIncr = true
|
||
|
kname = "#" + fmt.Sprint(count)
|
||
|
count++
|
||
|
}
|
||
|
|
||
|
lineRight := strings.TrimSpace(line[i+1:])
|
||
|
lineRightLength := len(lineRight)
|
||
|
firstChar := ""
|
||
|
if lineRightLength >= 2 {
|
||
|
firstChar = lineRight[0:1]
|
||
|
}
|
||
|
if firstChar == "`" {
|
||
|
valQuote = "`"
|
||
|
} else if firstChar == `"` {
|
||
|
if lineRightLength >= 3 && lineRight[0:3] == `"""` {
|
||
|
valQuote = `"""`
|
||
|
} else {
|
||
|
valQuote = `"`
|
||
|
}
|
||
|
} else if firstChar == `'` {
|
||
|
valQuote = `'`
|
||
|
}
|
||
|
|
||
|
if len(valQuote) > 0 {
|
||
|
qLen := len(valQuote)
|
||
|
pos := strings.LastIndex(lineRight[qLen:], valQuote)
|
||
|
// For multiple-line value check.
|
||
|
if pos == -1 {
|
||
|
if valQuote == `"` || valQuote == `'` {
|
||
|
return fmt.Errorf("error parsing line: single quote does not allow multiple-line value: %s", line)
|
||
|
}
|
||
|
|
||
|
val = lineRight[qLen:] + "\n"
|
||
|
val, err = checkMultipleLines(buf, line, val, valQuote)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
} else {
|
||
|
val = lineRight[qLen : pos+qLen]
|
||
|
}
|
||
|
} else {
|
||
|
val = strings.TrimSpace(cutComment(lineRight))
|
||
|
val, isEnd, err = checkContinuationLines(buf, val)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
k, err := section.NewKey(kname, val)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
k.isAutoIncr = isAutoIncr
|
||
|
if len(comments) > 0 {
|
||
|
k.Comment = comments
|
||
|
comments = ""
|
||
|
}
|
||
|
|
||
|
if isEnd {
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (f *File) reload(s dataSource) error {
|
||
|
r, err := s.ReadCloser()
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
defer r.Close()
|
||
|
|
||
|
return f.parse(r)
|
||
|
}
|
||
|
|
||
|
// Reload reloads and parses all data sources.
|
||
|
func (f *File) Reload() (err error) {
|
||
|
for _, s := range f.dataSources {
|
||
|
if err = f.reload(s); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// Append appends one or more data sources and reloads automatically.
|
||
|
func (f *File) Append(source interface{}, others ...interface{}) error {
|
||
|
ds, err := parseDataSource(source)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
f.dataSources = append(f.dataSources, ds)
|
||
|
for _, s := range others {
|
||
|
ds, err = parseDataSource(s)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
f.dataSources = append(f.dataSources, ds)
|
||
|
}
|
||
|
return f.Reload()
|
||
|
}
|
||
|
|
||
|
// WriteToIndent writes file content into io.Writer with given value indention.
|
||
|
func (f *File) WriteToIndent(w io.Writer, indent string) (n int64, err error) {
|
||
|
equalSign := "="
|
||
|
if PrettyFormat {
|
||
|
equalSign = " = "
|
||
|
}
|
||
|
|
||
|
// Use buffer to make sure target is safe until finish encoding.
|
||
|
buf := bytes.NewBuffer(nil)
|
||
|
for i, sname := range f.sectionList {
|
||
|
sec := f.Section(sname)
|
||
|
if len(sec.Comment) > 0 {
|
||
|
if sec.Comment[0] != '#' && sec.Comment[0] != ';' {
|
||
|
sec.Comment = "; " + sec.Comment
|
||
|
}
|
||
|
if _, err = buf.WriteString(sec.Comment + LineBreak); err != nil {
|
||
|
return 0, err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if i > 0 {
|
||
|
if _, err = buf.WriteString("[" + sname + "]" + LineBreak); err != nil {
|
||
|
return 0, err
|
||
|
}
|
||
|
} else {
|
||
|
// Write nothing if default section is empty.
|
||
|
if len(sec.keyList) == 0 {
|
||
|
continue
|
||
|
}
|
||
|
}
|
||
|
|
||
|
for _, kname := range sec.keyList {
|
||
|
key := sec.Key(kname)
|
||
|
if len(key.Comment) > 0 {
|
||
|
if len(indent) > 0 && sname != DEFAULT_SECTION {
|
||
|
buf.WriteString(indent)
|
||
|
}
|
||
|
if key.Comment[0] != '#' && key.Comment[0] != ';' {
|
||
|
key.Comment = "; " + key.Comment
|
||
|
}
|
||
|
if _, err = buf.WriteString(key.Comment + LineBreak); err != nil {
|
||
|
return 0, err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if len(indent) > 0 && sname != DEFAULT_SECTION {
|
||
|
buf.WriteString(indent)
|
||
|
}
|
||
|
|
||
|
switch {
|
||
|
case key.isAutoIncr:
|
||
|
kname = "-"
|
||
|
case strings.Contains(kname, "`") || strings.Contains(kname, `"`):
|
||
|
kname = `"""` + kname + `"""`
|
||
|
case strings.Contains(kname, `=`) || strings.Contains(kname, `:`):
|
||
|
kname = "`" + kname + "`"
|
||
|
}
|
||
|
|
||
|
val := key.value
|
||
|
// In case key value contains "\n", "`" or "\"".
|
||
|
if strings.Contains(val, "\n") || strings.Contains(val, "`") || strings.Contains(val, `"`) ||
|
||
|
strings.Contains(val, "#") {
|
||
|
val = `"""` + val + `"""`
|
||
|
}
|
||
|
if _, err = buf.WriteString(kname + equalSign + val + LineBreak); err != nil {
|
||
|
return 0, err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Put a line between sections.
|
||
|
if _, err = buf.WriteString(LineBreak); err != nil {
|
||
|
return 0, err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return buf.WriteTo(w)
|
||
|
}
|
||
|
|
||
|
// WriteTo writes file content into io.Writer.
|
||
|
func (f *File) WriteTo(w io.Writer) (int64, error) {
|
||
|
return f.WriteToIndent(w, "")
|
||
|
}
|
||
|
|
||
|
// SaveToIndent writes content to file system with given value indention.
|
||
|
func (f *File) SaveToIndent(filename, indent string) error {
|
||
|
// Note: Because we are truncating with os.Create,
|
||
|
// so it's safer to save to a temporary file location and rename afte done.
|
||
|
tmpPath := filename + "." + strconv.Itoa(time.Now().Nanosecond()) + ".tmp"
|
||
|
defer os.Remove(tmpPath)
|
||
|
|
||
|
fw, err := os.Create(tmpPath)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
if _, err = f.WriteToIndent(fw, indent); err != nil {
|
||
|
fw.Close()
|
||
|
return err
|
||
|
}
|
||
|
fw.Close()
|
||
|
|
||
|
// Remove old file and rename the new one.
|
||
|
os.Remove(filename)
|
||
|
return os.Rename(tmpPath, filename)
|
||
|
}
|
||
|
|
||
|
// SaveTo writes content to file system.
|
||
|
func (f *File) SaveTo(filename string) error {
|
||
|
return f.SaveToIndent(filename, "")
|
||
|
}
|