mirror of
https://github.com/astaxie/beego.git
synced 2025-01-11 05:27:13 +00:00
be544f963e
ParseBool returns the boolean value represented by the string. It accepts 1, 1.0, t, T, TRUE, true, True, YES, yes, Yes,Y, y, ON, on, On, 0, 0.0, f, F, FALSE, false, False, NO, no, No, N,n, OFF, off, Off. Any other value returns an error.
264 lines
6.4 KiB
Go
264 lines
6.4 KiB
Go
// Copyright 2014 beego Author. All Rights Reserved.
|
|
//
|
|
// 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 config
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"strings"
|
|
"sync"
|
|
)
|
|
|
|
// JSONConfig is a json config parser and implements Config interface.
|
|
type JSONConfig struct {
|
|
}
|
|
|
|
// Parse returns a ConfigContainer with parsed json config map.
|
|
func (js *JSONConfig) Parse(filename string) (Configer, error) {
|
|
file, err := os.Open(filename)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer file.Close()
|
|
content, err := ioutil.ReadAll(file)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return js.ParseData(content)
|
|
}
|
|
|
|
// ParseData returns a ConfigContainer with json string
|
|
func (js *JSONConfig) ParseData(data []byte) (Configer, error) {
|
|
x := &JSONConfigContainer{
|
|
data: make(map[string]interface{}),
|
|
}
|
|
err := json.Unmarshal(data, &x.data)
|
|
if err != nil {
|
|
var wrappingArray []interface{}
|
|
err2 := json.Unmarshal(data, &wrappingArray)
|
|
if err2 != nil {
|
|
return nil, err
|
|
}
|
|
x.data["rootArray"] = wrappingArray
|
|
}
|
|
return x, nil
|
|
}
|
|
|
|
// JSONConfigContainer A Config represents the json configuration.
|
|
// Only when get value, support key as section:name type.
|
|
type JSONConfigContainer struct {
|
|
data map[string]interface{}
|
|
sync.RWMutex
|
|
}
|
|
|
|
// Bool returns the boolean value for a given key.
|
|
func (c *JSONConfigContainer) Bool(key string) (bool, error) {
|
|
val := c.getData(key)
|
|
if val != nil {
|
|
return ParseBool(val)
|
|
}
|
|
return false, fmt.Errorf("not exist key: %q", key)
|
|
}
|
|
|
|
// DefaultBool return the bool value if has no error
|
|
// otherwise return the defaultval
|
|
func (c *JSONConfigContainer) DefaultBool(key string, defaultval bool) bool {
|
|
if v, err := c.Bool(key); err == nil {
|
|
return v
|
|
}
|
|
return defaultval
|
|
}
|
|
|
|
// Int returns the integer value for a given key.
|
|
func (c *JSONConfigContainer) Int(key string) (int, error) {
|
|
val := c.getData(key)
|
|
if val != nil {
|
|
if v, ok := val.(float64); ok {
|
|
return int(v), nil
|
|
}
|
|
return 0, errors.New("not int value")
|
|
}
|
|
return 0, errors.New("not exist key:" + key)
|
|
}
|
|
|
|
// DefaultInt returns the integer value for a given key.
|
|
// if err != nil return defaltval
|
|
func (c *JSONConfigContainer) DefaultInt(key string, defaultval int) int {
|
|
if v, err := c.Int(key); err == nil {
|
|
return v
|
|
}
|
|
return defaultval
|
|
}
|
|
|
|
// Int64 returns the int64 value for a given key.
|
|
func (c *JSONConfigContainer) Int64(key string) (int64, error) {
|
|
val := c.getData(key)
|
|
if val != nil {
|
|
if v, ok := val.(float64); ok {
|
|
return int64(v), nil
|
|
}
|
|
return 0, errors.New("not int64 value")
|
|
}
|
|
return 0, errors.New("not exist key:" + key)
|
|
}
|
|
|
|
// DefaultInt64 returns the int64 value for a given key.
|
|
// if err != nil return defaltval
|
|
func (c *JSONConfigContainer) DefaultInt64(key string, defaultval int64) int64 {
|
|
if v, err := c.Int64(key); err == nil {
|
|
return v
|
|
}
|
|
return defaultval
|
|
}
|
|
|
|
// Float returns the float value for a given key.
|
|
func (c *JSONConfigContainer) Float(key string) (float64, error) {
|
|
val := c.getData(key)
|
|
if val != nil {
|
|
if v, ok := val.(float64); ok {
|
|
return v, nil
|
|
}
|
|
return 0.0, errors.New("not float64 value")
|
|
}
|
|
return 0.0, errors.New("not exist key:" + key)
|
|
}
|
|
|
|
// DefaultFloat returns the float64 value for a given key.
|
|
// if err != nil return defaltval
|
|
func (c *JSONConfigContainer) DefaultFloat(key string, defaultval float64) float64 {
|
|
if v, err := c.Float(key); err == nil {
|
|
return v
|
|
}
|
|
return defaultval
|
|
}
|
|
|
|
// String returns the string value for a given key.
|
|
func (c *JSONConfigContainer) String(key string) string {
|
|
val := c.getData(key)
|
|
if val != nil {
|
|
if v, ok := val.(string); ok {
|
|
return v
|
|
}
|
|
}
|
|
return ""
|
|
}
|
|
|
|
// DefaultString returns the string value for a given key.
|
|
// if err != nil return defaltval
|
|
func (c *JSONConfigContainer) DefaultString(key string, defaultval string) string {
|
|
// TODO FIXME should not use "" to replace non existence
|
|
if v := c.String(key); v != "" {
|
|
return v
|
|
}
|
|
return defaultval
|
|
}
|
|
|
|
// Strings returns the []string value for a given key.
|
|
func (c *JSONConfigContainer) Strings(key string) []string {
|
|
stringVal := c.String(key)
|
|
if stringVal == "" {
|
|
return []string{}
|
|
}
|
|
return strings.Split(c.String(key), ";")
|
|
}
|
|
|
|
// DefaultStrings returns the []string value for a given key.
|
|
// if err != nil return defaltval
|
|
func (c *JSONConfigContainer) DefaultStrings(key string, defaultval []string) []string {
|
|
if v := c.Strings(key); len(v) > 0 {
|
|
return v
|
|
}
|
|
return defaultval
|
|
}
|
|
|
|
// GetSection returns map for the given section
|
|
func (c *JSONConfigContainer) GetSection(section string) (map[string]string, error) {
|
|
if v, ok := c.data[section]; ok {
|
|
return v.(map[string]string), nil
|
|
}
|
|
return nil, errors.New("nonexist section " + section)
|
|
}
|
|
|
|
// SaveConfigFile save the config into file
|
|
func (c *JSONConfigContainer) SaveConfigFile(filename string) (err error) {
|
|
// Write configuration file by filename.
|
|
f, err := os.Create(filename)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer f.Close()
|
|
b, err := json.MarshalIndent(c.data, "", " ")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = f.Write(b)
|
|
return err
|
|
}
|
|
|
|
// Set writes a new value for key.
|
|
func (c *JSONConfigContainer) Set(key, val string) error {
|
|
c.Lock()
|
|
defer c.Unlock()
|
|
c.data[key] = val
|
|
return nil
|
|
}
|
|
|
|
// DIY returns the raw value by a given key.
|
|
func (c *JSONConfigContainer) DIY(key string) (v interface{}, err error) {
|
|
val := c.getData(key)
|
|
if val != nil {
|
|
return val, nil
|
|
}
|
|
return nil, errors.New("not exist key")
|
|
}
|
|
|
|
// section.key or key
|
|
func (c *JSONConfigContainer) getData(key string) interface{} {
|
|
if len(key) == 0 {
|
|
return nil
|
|
}
|
|
|
|
c.RLock()
|
|
defer c.RUnlock()
|
|
|
|
sectionKeys := strings.Split(key, "::")
|
|
if len(sectionKeys) >= 2 {
|
|
curValue, ok := c.data[sectionKeys[0]]
|
|
if !ok {
|
|
return nil
|
|
}
|
|
for _, key := range sectionKeys[1:] {
|
|
if v, ok := curValue.(map[string]interface{}); ok {
|
|
if curValue, ok = v[key]; !ok {
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
return curValue
|
|
}
|
|
if v, ok := c.data[key]; ok {
|
|
return v
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func init() {
|
|
Register("json", &JSONConfig{})
|
|
}
|