2015-12-09 17:15:46 +00:00
|
|
|
// Disabling building of yaml support in cases where golang is 1.0 or 1.1
|
|
|
|
// as the encoding library is not implemented or supported.
|
|
|
|
|
2016-03-19 18:06:53 +00:00
|
|
|
// +build go1.2
|
2015-12-09 17:15:46 +00:00
|
|
|
|
|
|
|
package altsrc
|
2015-12-06 23:43:18 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"os"
|
|
|
|
|
|
|
|
"github.com/codegangsta/cli"
|
2015-12-09 17:15:46 +00:00
|
|
|
|
2015-12-06 23:43:18 +00:00
|
|
|
"gopkg.in/yaml.v2"
|
|
|
|
)
|
|
|
|
|
2015-12-09 17:15:46 +00:00
|
|
|
type yamlSourceContext struct {
|
2015-12-06 23:43:18 +00:00
|
|
|
FilePath string
|
|
|
|
}
|
|
|
|
|
2015-12-09 17:15:46 +00:00
|
|
|
// NewYamlSourceFromFile creates a new Yaml InputSourceContext from a filepath.
|
|
|
|
func NewYamlSourceFromFile(file string) (InputSourceContext, error) {
|
2016-03-19 18:06:53 +00:00
|
|
|
ysc := &yamlSourceContext{FilePath: file}
|
2016-04-29 16:33:59 +00:00
|
|
|
var results map[interface{}]interface{}
|
2016-03-19 18:06:53 +00:00
|
|
|
err := readCommandYaml(ysc.FilePath, &results)
|
2015-12-06 23:43:18 +00:00
|
|
|
if err != nil {
|
2016-03-19 18:06:53 +00:00
|
|
|
return nil, fmt.Errorf("Unable to load Yaml file '%s': inner error: \n'%v'", ysc.FilePath, err.Error())
|
2015-12-06 23:43:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return &MapInputSource{valueMap: results}, nil
|
|
|
|
}
|
|
|
|
|
2015-12-09 17:15:46 +00:00
|
|
|
// NewYamlSourceFromFlagFunc creates a new Yaml InputSourceContext from a provided flag name and source context.
|
2016-03-19 18:06:53 +00:00
|
|
|
func NewYamlSourceFromFlagFunc(flagFileName string) func(context *cli.Context) (InputSourceContext, error) {
|
|
|
|
return func(context *cli.Context) (InputSourceContext, error) {
|
2015-12-09 17:15:46 +00:00
|
|
|
filePath := context.String(flagFileName)
|
|
|
|
return NewYamlSourceFromFile(filePath)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-06 23:43:18 +00:00
|
|
|
func readCommandYaml(filePath string, container interface{}) (err error) {
|
|
|
|
b, err := loadDataFrom(filePath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = yaml.Unmarshal(b, container)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = nil
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func loadDataFrom(filePath string) ([]byte, error) {
|
|
|
|
u, err := url.Parse(filePath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if u.Host != "" { // i have a host, now do i support the scheme?
|
|
|
|
switch u.Scheme {
|
|
|
|
case "http", "https":
|
|
|
|
res, err := http.Get(filePath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return ioutil.ReadAll(res.Body)
|
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("scheme of %s is unsupported", filePath)
|
|
|
|
}
|
|
|
|
} else if u.Path != "" { // i dont have a host, but I have a path. I am a local file.
|
|
|
|
if _, notFoundFileErr := os.Stat(filePath); notFoundFileErr != nil {
|
|
|
|
return nil, fmt.Errorf("Cannot read from file: '%s' because it does not exist.", filePath)
|
|
|
|
}
|
|
|
|
return ioutil.ReadFile(filePath)
|
|
|
|
} else {
|
|
|
|
return nil, fmt.Errorf("unable to determine how to load from path %s", filePath)
|
|
|
|
}
|
|
|
|
}
|