2017-06-25 21:45:22 +00:00
|
|
|
// +-------------------------------------------------------------------------
|
|
|
|
// | Copyright (C) 2016 Yunify, Inc.
|
|
|
|
// +-------------------------------------------------------------------------
|
|
|
|
// | Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// | you may not use this work except in compliance with the License.
|
|
|
|
// | You may obtain a copy of the License in the LICENSE file, or 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 (
|
|
|
|
"errors"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"strings"
|
2018-01-16 13:20:59 +00:00
|
|
|
"time"
|
2017-06-25 21:45:22 +00:00
|
|
|
|
2018-01-16 13:20:59 +00:00
|
|
|
"gopkg.in/yaml.v2"
|
2017-06-25 21:45:22 +00:00
|
|
|
|
|
|
|
"github.com/yunify/qingstor-sdk-go/logger"
|
|
|
|
)
|
|
|
|
|
|
|
|
// A Config stores a configuration of this sdk.
|
|
|
|
type Config struct {
|
|
|
|
AccessKeyID string `yaml:"access_key_id"`
|
|
|
|
SecretAccessKey string `yaml:"secret_access_key"`
|
|
|
|
|
|
|
|
Host string `yaml:"host"`
|
|
|
|
Port int `yaml:"port"`
|
|
|
|
Protocol string `yaml:"protocol"`
|
|
|
|
ConnectionRetries int `yaml:"connection_retries"`
|
|
|
|
|
|
|
|
AdditionalUserAgent string `yaml:"additional_user_agent"`
|
|
|
|
|
|
|
|
LogLevel string `yaml:"log_level"`
|
|
|
|
|
|
|
|
Connection *http.Client
|
|
|
|
}
|
|
|
|
|
|
|
|
// New create a Config with given AccessKeyID and SecretAccessKey.
|
2018-01-16 13:20:59 +00:00
|
|
|
func New(accessKeyID, secretAccessKey string) (c *Config, err error) {
|
|
|
|
c, err = NewDefault()
|
2017-06-25 21:45:22 +00:00
|
|
|
if err != nil {
|
2018-01-16 13:20:59 +00:00
|
|
|
c = nil
|
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
|
2018-01-16 13:20:59 +00:00
|
|
|
c.AccessKeyID = accessKeyID
|
|
|
|
c.SecretAccessKey = secretAccessKey
|
2017-06-25 21:45:22 +00:00
|
|
|
|
2018-01-16 13:20:59 +00:00
|
|
|
c.Connection = &http.Client{
|
|
|
|
Timeout: time.Minute,
|
|
|
|
}
|
2017-06-25 21:45:22 +00:00
|
|
|
|
2018-01-16 13:20:59 +00:00
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewDefault create a Config with default configuration.
|
2018-01-16 13:20:59 +00:00
|
|
|
func NewDefault() (c *Config, err error) {
|
|
|
|
c = &Config{}
|
|
|
|
err = c.LoadDefaultConfig()
|
2017-06-25 21:45:22 +00:00
|
|
|
if err != nil {
|
2018-01-16 13:20:59 +00:00
|
|
|
c = nil
|
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
|
2018-01-16 13:20:59 +00:00
|
|
|
c.Connection = &http.Client{
|
|
|
|
Timeout: time.Minute,
|
|
|
|
}
|
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check checks the configuration.
|
2018-01-16 13:20:59 +00:00
|
|
|
func (c *Config) Check() (err error) {
|
2017-06-25 21:45:22 +00:00
|
|
|
if c.AccessKeyID == "" {
|
2018-01-16 13:20:59 +00:00
|
|
|
err = errors.New("access key ID not specified")
|
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
if c.SecretAccessKey == "" {
|
2018-01-16 13:20:59 +00:00
|
|
|
err = errors.New("secret access key not specified")
|
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if c.Host == "" {
|
2018-01-16 13:20:59 +00:00
|
|
|
err = errors.New("server host not specified")
|
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
if c.Port <= 0 {
|
2018-01-16 13:20:59 +00:00
|
|
|
err = errors.New("server port not specified")
|
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
if c.Protocol == "" {
|
2018-01-16 13:20:59 +00:00
|
|
|
err = errors.New("server protocol not specified")
|
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if c.AdditionalUserAgent != "" {
|
|
|
|
for _, x := range c.AdditionalUserAgent {
|
|
|
|
// Allow space(32) to ~(126) in ASCII Table, exclude "(34).
|
|
|
|
if int(x) < 32 || int(x) > 126 || int(x) == 32 || int(x) == 34 {
|
2018-01-16 13:20:59 +00:00
|
|
|
err = errors.New("additional User-Agent contains characters that not allowed")
|
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-16 13:20:59 +00:00
|
|
|
err = logger.CheckLevel(c.LogLevel)
|
2017-06-25 21:45:22 +00:00
|
|
|
if err != nil {
|
2018-01-16 13:20:59 +00:00
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
|
2018-01-16 13:20:59 +00:00
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// LoadDefaultConfig loads the default configuration for Config.
|
|
|
|
// It returns error if yaml decode failed.
|
2018-01-16 13:20:59 +00:00
|
|
|
func (c *Config) LoadDefaultConfig() (err error) {
|
|
|
|
err = yaml.Unmarshal([]byte(DefaultConfigFileContent), c)
|
2017-06-25 21:45:22 +00:00
|
|
|
if err != nil {
|
2018-01-16 13:20:59 +00:00
|
|
|
logger.Errorf(nil, "Config parse error, %v.", err)
|
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
logger.SetLevel(c.LogLevel)
|
2018-01-16 13:20:59 +00:00
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// LoadUserConfig loads user configuration in ~/.qingstor/config.yaml for Config.
|
|
|
|
// It returns error if file not found.
|
2018-01-16 13:20:59 +00:00
|
|
|
func (c *Config) LoadUserConfig() (err error) {
|
|
|
|
_, err = os.Stat(GetUserConfigFilePath())
|
2017-06-25 21:45:22 +00:00
|
|
|
if err != nil {
|
2018-01-16 13:20:59 +00:00
|
|
|
logger.Warnf(nil, "Installing default config file to %s.", GetUserConfigFilePath())
|
2017-06-25 21:45:22 +00:00
|
|
|
InstallDefaultUserConfig()
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.LoadConfigFromFilePath(GetUserConfigFilePath())
|
|
|
|
}
|
|
|
|
|
|
|
|
// LoadConfigFromFilePath loads configuration from a specified local path.
|
|
|
|
// It returns error if file not found or yaml decode failed.
|
2018-01-16 13:20:59 +00:00
|
|
|
func (c *Config) LoadConfigFromFilePath(filePath string) (err error) {
|
|
|
|
if strings.Index(filePath, "~/") == 0 {
|
|
|
|
filePath = strings.Replace(filePath, "~/", getHome()+"/", 1)
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
|
2018-01-16 13:20:59 +00:00
|
|
|
yamlString, err := ioutil.ReadFile(filePath)
|
2017-06-25 21:45:22 +00:00
|
|
|
if err != nil {
|
2018-01-16 13:20:59 +00:00
|
|
|
logger.Errorf(nil, "File not found: %s.", filePath)
|
2017-06-25 21:45:22 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.LoadConfigFromContent(yamlString)
|
|
|
|
}
|
|
|
|
|
|
|
|
// LoadConfigFromContent loads configuration from a given byte slice.
|
|
|
|
// It returns error if yaml decode failed.
|
2018-01-16 13:20:59 +00:00
|
|
|
func (c *Config) LoadConfigFromContent(content []byte) (err error) {
|
2017-06-25 21:45:22 +00:00
|
|
|
c.LoadDefaultConfig()
|
|
|
|
|
2018-01-16 13:20:59 +00:00
|
|
|
err = yaml.Unmarshal(content, c)
|
2017-06-25 21:45:22 +00:00
|
|
|
if err != nil {
|
2018-01-16 13:20:59 +00:00
|
|
|
logger.Errorf(nil, "Config parse error, %v.", err)
|
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
err = c.Check()
|
|
|
|
if err != nil {
|
2018-01-16 13:20:59 +00:00
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
logger.SetLevel(c.LogLevel)
|
2018-01-16 13:20:59 +00:00
|
|
|
return
|
2017-06-25 21:45:22 +00:00
|
|
|
}
|