126 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			126 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package beego
 | |
| 
 | |
| import (
 | |
| 	"bufio"
 | |
| 	"bytes"
 | |
| 	"errors"
 | |
| 	"io"
 | |
| 	"os"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"sync"
 | |
| 	"unicode"
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	bComment = []byte{'#'}
 | |
| 	bEmpty   = []byte{}
 | |
| 	bEqual   = []byte{'='}
 | |
| 	bDQuote  = []byte{'"'}
 | |
| )
 | |
| 
 | |
| // A Config represents the configuration.
 | |
| type Config struct {
 | |
| 	filename string
 | |
| 	comment  map[int][]string  // id: []{comment, key...}; id 1 is for main comment.
 | |
| 	data     map[string]string // key: value
 | |
| 	offset   map[string]int64  // key: offset; for editing.
 | |
| 	sync.RWMutex
 | |
| }
 | |
| 
 | |
| // ParseFile creates a new Config and parses the file configuration from the
 | |
| // named file.
 | |
| func LoadConfig(name string) (*Config, error) {
 | |
| 	file, err := os.Open(name)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	cfg := &Config{
 | |
| 		file.Name(),
 | |
| 		make(map[int][]string),
 | |
| 		make(map[string]string),
 | |
| 		make(map[string]int64),
 | |
| 		sync.RWMutex{},
 | |
| 	}
 | |
| 	cfg.Lock()
 | |
| 	defer cfg.Unlock()
 | |
| 	defer file.Close()
 | |
| 
 | |
| 	var comment bytes.Buffer
 | |
| 	buf := bufio.NewReader(file)
 | |
| 
 | |
| 	for nComment, off := 0, int64(1); ; {
 | |
| 		line, _, err := buf.ReadLine()
 | |
| 		if err == io.EOF {
 | |
| 			break
 | |
| 		}
 | |
| 		if bytes.Equal(line, bEmpty) {
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		off += int64(len(line))
 | |
| 
 | |
| 		if bytes.HasPrefix(line, bComment) {
 | |
| 			line = bytes.TrimLeft(line, "#")
 | |
| 			line = bytes.TrimLeftFunc(line, unicode.IsSpace)
 | |
| 			comment.Write(line)
 | |
| 			comment.WriteByte('\n')
 | |
| 			continue
 | |
| 		}
 | |
| 		if comment.Len() != 0 {
 | |
| 			cfg.comment[nComment] = []string{comment.String()}
 | |
| 			comment.Reset()
 | |
| 			nComment++
 | |
| 		}
 | |
| 
 | |
| 		val := bytes.SplitN(line, bEqual, 2)
 | |
| 		if bytes.HasPrefix(val[1], bDQuote) {
 | |
| 			val[1] = bytes.Trim(val[1], `"`)
 | |
| 		}
 | |
| 
 | |
| 		key := strings.TrimSpace(string(val[0]))
 | |
| 		cfg.comment[nComment-1] = append(cfg.comment[nComment-1], key)
 | |
| 		cfg.data[key] = strings.TrimSpace(string(val[1]))
 | |
| 		cfg.offset[key] = off
 | |
| 	}
 | |
| 	return cfg, nil
 | |
| }
 | |
| 
 | |
| // Bool returns the boolean value for a given key.
 | |
| func (c *Config) Bool(key string) (bool, error) {
 | |
| 	return strconv.ParseBool(c.data[key])
 | |
| }
 | |
| 
 | |
| // Int returns the integer value for a given key.
 | |
| func (c *Config) Int(key string) (int, error) {
 | |
| 	return strconv.Atoi(c.data[key])
 | |
| }
 | |
| 
 | |
| func (c *Config) Int64(key string) (int64, error) {
 | |
| 	return strconv.ParseInt(c.data[key], 10, 64)
 | |
| }
 | |
| 
 | |
| // Float returns the float value for a given key.
 | |
| func (c *Config) Float(key string) (float64, error) {
 | |
| 	return strconv.ParseFloat(c.data[key], 64)
 | |
| }
 | |
| 
 | |
| // String returns the string value for a given key.
 | |
| func (c *Config) String(key string) string {
 | |
| 	return c.data[key]
 | |
| }
 | |
| 
 | |
| // WriteValue writes a new value for key.
 | |
| func (c *Config) SetValue(key, value string) error {
 | |
| 	c.Lock()
 | |
| 	defer c.Unlock()
 | |
| 
 | |
| 	if _, found := c.data[key]; !found {
 | |
| 		return errors.New("key not found: " + key)
 | |
| 	}
 | |
| 
 | |
| 	c.data[key] = value
 | |
| 	return nil
 | |
| }
 |