forked from TrueCloudLab/rclone
b3f55d6bda
This fixes listing sections just after creation which means the rclone config list will have all the keys in now.
555 lines
15 KiB
Go
555 lines
15 KiB
Go
// Copyright 2013 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 goconfig is a fully functional and comments-support configuration file(.ini) parser.
|
|
package goconfig
|
|
|
|
import (
|
|
"fmt"
|
|
"regexp"
|
|
"runtime"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
)
|
|
|
|
const (
|
|
// Default section name.
|
|
DEFAULT_SECTION = "DEFAULT"
|
|
// Maximum allowed depth when recursively substituing variable names.
|
|
_DEPTH_VALUES = 200
|
|
)
|
|
|
|
type ParseError int
|
|
|
|
const (
|
|
ERR_SECTION_NOT_FOUND ParseError = iota + 1
|
|
ERR_KEY_NOT_FOUND
|
|
ERR_BLANK_SECTION_NAME
|
|
ERR_COULD_NOT_PARSE
|
|
)
|
|
|
|
var LineBreak = "\n"
|
|
|
|
// Variable regexp pattern: %(variable)s
|
|
var varPattern = regexp.MustCompile(`%\(([^\)]+)\)s`)
|
|
|
|
func init() {
|
|
if runtime.GOOS == "windows" {
|
|
LineBreak = "\r\n"
|
|
}
|
|
}
|
|
|
|
// A ConfigFile represents a INI formar configuration file.
|
|
type ConfigFile struct {
|
|
lock sync.RWMutex // Go map is not safe.
|
|
fileNames []string // Support mutil-files.
|
|
data map[string]map[string]string // Section -> key : value
|
|
|
|
// Lists can keep sections and keys in order.
|
|
sectionList []string // Section name list.
|
|
keyList map[string][]string // Section -> Key name list
|
|
|
|
sectionComments map[string]string // Sections comments.
|
|
keyComments map[string]map[string]string // Keys comments.
|
|
BlockMode bool // Indicates whether use lock or not.
|
|
}
|
|
|
|
// newConfigFile creates an empty configuration representation.
|
|
func newConfigFile(fileNames []string) *ConfigFile {
|
|
c := new(ConfigFile)
|
|
c.fileNames = fileNames
|
|
c.data = make(map[string]map[string]string)
|
|
c.keyList = make(map[string][]string)
|
|
c.sectionComments = make(map[string]string)
|
|
c.keyComments = make(map[string]map[string]string)
|
|
c.BlockMode = true
|
|
return c
|
|
}
|
|
|
|
// SetValue adds a new section-key-value to the configuration.
|
|
// It returns true if the key and value were inserted,
|
|
// or returns false if the value was overwritten.
|
|
// If the section does not exist in advance, it will be created.
|
|
func (c *ConfigFile) SetValue(section, key, value string) bool {
|
|
// Blank section name represents DEFAULT section.
|
|
if len(section) == 0 {
|
|
section = DEFAULT_SECTION
|
|
}
|
|
if len(key) == 0 {
|
|
return false
|
|
}
|
|
|
|
if c.BlockMode {
|
|
c.lock.Lock()
|
|
defer c.lock.Unlock()
|
|
}
|
|
|
|
// Check if section exists.
|
|
if _, ok := c.data[section]; !ok {
|
|
// Execute add operation.
|
|
c.data[section] = make(map[string]string)
|
|
// Append section to list.
|
|
c.sectionList = append(c.sectionList, section)
|
|
}
|
|
|
|
// Check if key exists.
|
|
_, ok := c.data[section][key]
|
|
c.data[section][key] = value
|
|
if !ok {
|
|
// If not exists, append to key list.
|
|
c.keyList[section] = append(c.keyList[section], key)
|
|
}
|
|
return !ok
|
|
}
|
|
|
|
// DeleteKey deletes the key in given section.
|
|
// It returns true if the key was deleted,
|
|
// or returns false if the section or key didn't exist.
|
|
func (c *ConfigFile) DeleteKey(section, key string) bool {
|
|
// Blank section name represents DEFAULT section.
|
|
if len(section) == 0 {
|
|
section = DEFAULT_SECTION
|
|
}
|
|
|
|
if c.BlockMode {
|
|
c.lock.Lock()
|
|
defer c.lock.Unlock()
|
|
}
|
|
|
|
// Check if section exists.
|
|
if _, ok := c.data[section]; !ok {
|
|
return false
|
|
}
|
|
|
|
// Check if key exists.
|
|
if _, ok := c.data[section][key]; ok {
|
|
delete(c.data[section], key)
|
|
// Remove comments of key.
|
|
c.SetKeyComments(section, key, "")
|
|
// Get index of key.
|
|
i := 0
|
|
for _, keyName := range c.keyList[section] {
|
|
if keyName == key {
|
|
break
|
|
}
|
|
i++
|
|
}
|
|
// Remove from key list.
|
|
c.keyList[section] = append(c.keyList[section][:i], c.keyList[section][i+1:]...)
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
// GetValue returns the value of key available in the given section.
|
|
// If the value needs to be unfolded
|
|
// (see e.g. %(google)s example in the GoConfig_test.go),
|
|
// then String does this unfolding automatically, up to
|
|
// _DEPTH_VALUES number of iterations.
|
|
// It returns an error and empty string value if the section does not exist,
|
|
// or key does not exist in DEFAULT and current sections.
|
|
func (c *ConfigFile) GetValue(section, key string) (string, error) {
|
|
if c.BlockMode {
|
|
c.lock.RLock()
|
|
defer c.lock.RUnlock()
|
|
}
|
|
|
|
// Blank section name represents DEFAULT section.
|
|
if len(section) == 0 {
|
|
section = DEFAULT_SECTION
|
|
}
|
|
|
|
// Check if section exists
|
|
if _, ok := c.data[section]; !ok {
|
|
// Section does not exist.
|
|
return "", getError{ERR_SECTION_NOT_FOUND, section}
|
|
}
|
|
|
|
// Section exists.
|
|
// Check if key exists or empty value.
|
|
value, ok := c.data[section][key]
|
|
if !ok {
|
|
// Check if it is a sub-section.
|
|
if i := strings.LastIndex(section, "."); i > -1 {
|
|
return c.GetValue(section[:i], key)
|
|
}
|
|
|
|
// Return empty value.
|
|
return "", getError{ERR_KEY_NOT_FOUND, key}
|
|
}
|
|
|
|
// Key exists.
|
|
var i int
|
|
for i = 0; i < _DEPTH_VALUES; i++ {
|
|
vr := varPattern.FindString(value)
|
|
if len(vr) == 0 {
|
|
break
|
|
}
|
|
|
|
// Take off leading '%(' and trailing ')s'.
|
|
noption := strings.TrimLeft(vr, "%(")
|
|
noption = strings.TrimRight(noption, ")s")
|
|
|
|
// Search variable in default section.
|
|
nvalue, err := c.GetValue(DEFAULT_SECTION, noption)
|
|
if err != nil && section != DEFAULT_SECTION {
|
|
// Search in the same section.
|
|
if _, ok := c.data[section][noption]; ok {
|
|
nvalue = c.data[section][noption]
|
|
}
|
|
}
|
|
|
|
// Substitute by new value and take off leading '%(' and trailing ')s'.
|
|
value = strings.Replace(value, vr, nvalue, -1)
|
|
}
|
|
return value, nil
|
|
}
|
|
|
|
// Bool returns bool type value.
|
|
func (c *ConfigFile) Bool(section, key string) (bool, error) {
|
|
value, err := c.GetValue(section, key)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return strconv.ParseBool(value)
|
|
}
|
|
|
|
// Float64 returns float64 type value.
|
|
func (c *ConfigFile) Float64(section, key string) (float64, error) {
|
|
value, err := c.GetValue(section, key)
|
|
if err != nil {
|
|
return 0.0, err
|
|
}
|
|
return strconv.ParseFloat(value, 64)
|
|
}
|
|
|
|
// Int returns int type value.
|
|
func (c *ConfigFile) Int(section, key string) (int, error) {
|
|
value, err := c.GetValue(section, key)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return strconv.Atoi(value)
|
|
}
|
|
|
|
// Int64 returns int64 type value.
|
|
func (c *ConfigFile) Int64(section, key string) (int64, error) {
|
|
value, err := c.GetValue(section, key)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
return strconv.ParseInt(value, 10, 64)
|
|
}
|
|
|
|
// MustValue always returns value without error.
|
|
// It returns empty string if error occurs, or the default value if given.
|
|
func (c *ConfigFile) MustValue(section, key string, defaultVal ...string) string {
|
|
val, err := c.GetValue(section, key)
|
|
if len(defaultVal) > 0 && (err != nil || len(val) == 0) {
|
|
return defaultVal[0]
|
|
}
|
|
return val
|
|
}
|
|
|
|
// MustValue always returns value without error,
|
|
// It returns empty string if error occurs, or the default value if given,
|
|
// and a bool value indicates whether default value is returned.
|
|
func (c *ConfigFile) MustValueSet(section, key string, defaultVal ...string) (string, bool) {
|
|
val, err := c.GetValue(section, key)
|
|
if len(defaultVal) > 0 && (err != nil || len(val) == 0) {
|
|
c.SetValue(section, key, defaultVal[0])
|
|
return defaultVal[0], true
|
|
}
|
|
return val, false
|
|
}
|
|
|
|
// MustValueRange always returns value without error,
|
|
// it returns default value if error occurs or doesn't fit into range.
|
|
func (c *ConfigFile) MustValueRange(section, key, defaultVal string, candidates []string) string {
|
|
val, err := c.GetValue(section, key)
|
|
if err != nil || len(val) == 0 {
|
|
return defaultVal
|
|
}
|
|
|
|
for _, cand := range candidates {
|
|
if val == cand {
|
|
return val
|
|
}
|
|
}
|
|
return defaultVal
|
|
}
|
|
|
|
// MustValueArray always returns value array without error,
|
|
// it returns empty array if error occurs, split by delimiter otherwise.
|
|
func (c *ConfigFile) MustValueArray(section, key, delim string) []string {
|
|
val, err := c.GetValue(section, key)
|
|
if err != nil || len(val) == 0 {
|
|
return []string{}
|
|
}
|
|
|
|
vals := strings.Split(val, delim)
|
|
for i := range vals {
|
|
vals[i] = strings.TrimSpace(vals[i])
|
|
}
|
|
return vals
|
|
}
|
|
|
|
// MustBool always returns value without error,
|
|
// it returns false if error occurs.
|
|
func (c *ConfigFile) MustBool(section, key string, defaultVal ...bool) bool {
|
|
val, err := c.Bool(section, key)
|
|
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 (c *ConfigFile) MustFloat64(section, key string, defaultVal ...float64) float64 {
|
|
value, err := c.Float64(section, key)
|
|
if len(defaultVal) > 0 && err != nil {
|
|
return defaultVal[0]
|
|
}
|
|
return value
|
|
}
|
|
|
|
// MustInt always returns value without error,
|
|
// it returns 0 if error occurs.
|
|
func (c *ConfigFile) MustInt(section, key string, defaultVal ...int) int {
|
|
value, err := c.Int(section, key)
|
|
if len(defaultVal) > 0 && err != nil {
|
|
return defaultVal[0]
|
|
}
|
|
return value
|
|
}
|
|
|
|
// MustInt64 always returns value without error,
|
|
// it returns 0 if error occurs.
|
|
func (c *ConfigFile) MustInt64(section, key string, defaultVal ...int64) int64 {
|
|
value, err := c.Int64(section, key)
|
|
if len(defaultVal) > 0 && err != nil {
|
|
return defaultVal[0]
|
|
}
|
|
return value
|
|
}
|
|
|
|
// GetSectionList returns the list of all sections
|
|
// in the same order in the file.
|
|
func (c *ConfigFile) GetSectionList() []string {
|
|
list := make([]string, len(c.sectionList))
|
|
copy(list, c.sectionList)
|
|
return list
|
|
}
|
|
|
|
// GetKeyList returns the list of all keys in give section
|
|
// in the same order in the file.
|
|
// It returns nil if given section does not exist.
|
|
func (c *ConfigFile) GetKeyList(section string) []string {
|
|
// Blank section name represents DEFAULT section.
|
|
if len(section) == 0 {
|
|
section = DEFAULT_SECTION
|
|
}
|
|
|
|
if c.BlockMode {
|
|
c.lock.RLock()
|
|
defer c.lock.RUnlock()
|
|
}
|
|
|
|
// Check if section exists.
|
|
if _, ok := c.data[section]; !ok {
|
|
return nil
|
|
}
|
|
|
|
// Non-default section has a blank key as section keeper.
|
|
list := make([]string, 0, len(c.keyList[section]))
|
|
for _, key := range c.keyList[section] {
|
|
if key != " " {
|
|
list = append(list, key)
|
|
}
|
|
}
|
|
return list
|
|
}
|
|
|
|
// DeleteSection deletes the entire section by given name.
|
|
// It returns true if the section was deleted, and false if the section didn't exist.
|
|
func (c *ConfigFile) DeleteSection(section string) bool {
|
|
// Blank section name represents DEFAULT section.
|
|
if len(section) == 0 {
|
|
section = DEFAULT_SECTION
|
|
}
|
|
|
|
if c.BlockMode {
|
|
c.lock.Lock()
|
|
defer c.lock.Unlock()
|
|
}
|
|
|
|
// Check if section exists.
|
|
if _, ok := c.data[section]; !ok {
|
|
return false
|
|
}
|
|
|
|
delete(c.data, section)
|
|
// Remove comments of section.
|
|
c.SetSectionComments(section, "")
|
|
// Get index of section.
|
|
i := 0
|
|
for _, secName := range c.sectionList {
|
|
if secName == section {
|
|
break
|
|
}
|
|
i++
|
|
}
|
|
// Remove from section and key list.
|
|
c.sectionList = append(c.sectionList[:i], c.sectionList[i+1:]...)
|
|
delete(c.keyList, section)
|
|
return true
|
|
}
|
|
|
|
// GetSection returns key-value pairs in given section.
|
|
// If section does not exist, returns nil and error.
|
|
func (c *ConfigFile) GetSection(section string) (map[string]string, error) {
|
|
// Blank section name represents DEFAULT section.
|
|
if len(section) == 0 {
|
|
section = DEFAULT_SECTION
|
|
}
|
|
|
|
if c.BlockMode {
|
|
c.lock.Lock()
|
|
defer c.lock.Unlock()
|
|
}
|
|
|
|
// Check if section exists.
|
|
if _, ok := c.data[section]; !ok {
|
|
// Section does not exist.
|
|
return nil, getError{ERR_SECTION_NOT_FOUND, section}
|
|
}
|
|
|
|
// Remove pre-defined key.
|
|
secMap := c.data[section]
|
|
delete(c.data[section], " ")
|
|
|
|
// Section exists.
|
|
return secMap, nil
|
|
}
|
|
|
|
// SetSectionComments adds new section comments to the configuration.
|
|
// If comments are empty(0 length), it will remove its section comments!
|
|
// It returns true if the comments were inserted or removed,
|
|
// or returns false if the comments were overwritten.
|
|
func (c *ConfigFile) SetSectionComments(section, comments string) bool {
|
|
// Blank section name represents DEFAULT section.
|
|
if len(section) == 0 {
|
|
section = DEFAULT_SECTION
|
|
}
|
|
|
|
if len(comments) == 0 {
|
|
if _, ok := c.sectionComments[section]; ok {
|
|
delete(c.sectionComments, section)
|
|
}
|
|
|
|
// Not exists can be seen as remove.
|
|
return true
|
|
}
|
|
|
|
// Check if comments exists.
|
|
_, ok := c.sectionComments[section]
|
|
if comments[0] != '#' && comments[0] != ';' {
|
|
comments = "; " + comments
|
|
}
|
|
c.sectionComments[section] = comments
|
|
return !ok
|
|
}
|
|
|
|
// SetKeyComments adds new section-key comments to the configuration.
|
|
// If comments are empty(0 length), it will remove its section-key comments!
|
|
// It returns true if the comments were inserted or removed,
|
|
// or returns false if the comments were overwritten.
|
|
// If the section does not exist in advance, it is created.
|
|
func (c *ConfigFile) SetKeyComments(section, key, comments string) bool {
|
|
// Blank section name represents DEFAULT section.
|
|
if len(section) == 0 {
|
|
section = DEFAULT_SECTION
|
|
}
|
|
|
|
// Check if section exists.
|
|
if _, ok := c.keyComments[section]; ok {
|
|
if len(comments) == 0 {
|
|
if _, ok := c.keyComments[section][key]; ok {
|
|
delete(c.keyComments[section], key)
|
|
}
|
|
|
|
// Not exists can be seen as remove.
|
|
return true
|
|
}
|
|
} else {
|
|
if len(comments) == 0 {
|
|
// Not exists can be seen as remove.
|
|
return true
|
|
} else {
|
|
// Execute add operation.
|
|
c.keyComments[section] = make(map[string]string)
|
|
}
|
|
}
|
|
|
|
// Check if key exists.
|
|
_, ok := c.keyComments[section][key]
|
|
if comments[0] != '#' && comments[0] != ';' {
|
|
comments = "; " + comments
|
|
}
|
|
c.keyComments[section][key] = comments
|
|
return !ok
|
|
}
|
|
|
|
// GetSectionComments returns the comments in the given section.
|
|
// It returns an empty string(0 length) if the comments do not exist.
|
|
func (c *ConfigFile) GetSectionComments(section string) (comments string) {
|
|
// Blank section name represents DEFAULT section.
|
|
if len(section) == 0 {
|
|
section = DEFAULT_SECTION
|
|
}
|
|
return c.sectionComments[section]
|
|
}
|
|
|
|
// GetKeyComments returns the comments of key in the given section.
|
|
// It returns an empty string(0 length) if the comments do not exist.
|
|
func (c *ConfigFile) GetKeyComments(section, key string) (comments string) {
|
|
// Blank section name represents DEFAULT section.
|
|
if len(section) == 0 {
|
|
section = DEFAULT_SECTION
|
|
}
|
|
|
|
if _, ok := c.keyComments[section]; ok {
|
|
return c.keyComments[section][key]
|
|
}
|
|
return ""
|
|
}
|
|
|
|
// getError occurs when get value in configuration file with invalid parameter.
|
|
type getError struct {
|
|
Reason ParseError
|
|
Name string
|
|
}
|
|
|
|
// Error implements Error interface.
|
|
func (err getError) Error() string {
|
|
switch err.Reason {
|
|
case ERR_SECTION_NOT_FOUND:
|
|
return fmt.Sprintf("section '%s' not found", err.Name)
|
|
case ERR_KEY_NOT_FOUND:
|
|
return fmt.Sprintf("key '%s' not found", err.Name)
|
|
}
|
|
return "invalid get error"
|
|
}
|