2014-11-10 17:09:32 +00:00
|
|
|
/*
|
|
|
|
Copyright 2014 Google Inc. 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.
|
|
|
|
*/
|
|
|
|
|
2015-02-05 00:14:48 +00:00
|
|
|
package util
|
2014-11-10 17:09:32 +00:00
|
|
|
|
|
|
|
import (
|
2015-01-15 03:29:13 +00:00
|
|
|
"encoding/json"
|
2014-11-10 17:09:32 +00:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
2014-12-10 21:48:48 +00:00
|
|
|
"time"
|
2014-11-10 17:09:32 +00:00
|
|
|
|
2015-01-15 03:29:13 +00:00
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/api"
|
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/runtime"
|
2015-02-23 19:47:06 +00:00
|
|
|
|
|
|
|
"github.com/evanphx/json-patch"
|
2014-11-10 17:09:32 +00:00
|
|
|
"github.com/golang/glog"
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
)
|
|
|
|
|
2015-02-05 00:14:48 +00:00
|
|
|
func checkErr(err error) {
|
|
|
|
if err != nil {
|
2015-02-10 18:02:16 +00:00
|
|
|
glog.FatalDepth(1, err.Error())
|
2015-02-05 00:14:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func usageError(cmd *cobra.Command, format string, args ...interface{}) {
|
|
|
|
glog.Errorf(format, args...)
|
|
|
|
glog.Errorf("See '%s -h' for help.", cmd.CommandPath())
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
|
2014-11-10 17:09:32 +00:00
|
|
|
func GetFlagString(cmd *cobra.Command, flag string) string {
|
|
|
|
f := cmd.Flags().Lookup(flag)
|
|
|
|
if f == nil {
|
|
|
|
glog.Fatalf("Flag accessed but not defined for command %s: %s", cmd.Name(), flag)
|
|
|
|
}
|
|
|
|
return f.Value.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetFlagBool(cmd *cobra.Command, flag string) bool {
|
|
|
|
f := cmd.Flags().Lookup(flag)
|
|
|
|
if f == nil {
|
|
|
|
glog.Fatalf("Flag accessed but not defined for command %s: %s", cmd.Name(), flag)
|
|
|
|
}
|
|
|
|
// Caseless compare.
|
|
|
|
if strings.ToLower(f.Value.String()) == "true" {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// Assumes the flag has a default value.
|
|
|
|
func GetFlagInt(cmd *cobra.Command, flag string) int {
|
|
|
|
f := cmd.Flags().Lookup(flag)
|
|
|
|
if f == nil {
|
|
|
|
glog.Fatalf("Flag accessed but not defined for command %s: %s", cmd.Name(), flag)
|
|
|
|
}
|
|
|
|
v, err := strconv.Atoi(f.Value.String())
|
|
|
|
// This is likely not a sufficiently friendly error message, but cobra
|
|
|
|
// should prevent non-integer values from reaching here.
|
|
|
|
checkErr(err)
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
2014-12-10 21:48:48 +00:00
|
|
|
func GetFlagDuration(cmd *cobra.Command, flag string) time.Duration {
|
|
|
|
f := cmd.Flags().Lookup(flag)
|
|
|
|
if f == nil {
|
|
|
|
glog.Fatalf("Flag accessed but not defined for command %s: %s", cmd.Name(), flag)
|
|
|
|
}
|
|
|
|
v, err := time.ParseDuration(f.Value.String())
|
|
|
|
checkErr(err)
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
2014-11-10 17:09:32 +00:00
|
|
|
// ReadConfigData reads the bytes from the specified filesytem or network
|
|
|
|
// location or from stdin if location == "-".
|
2014-12-27 21:48:27 +00:00
|
|
|
// TODO: replace with resource.Builder
|
2014-11-10 17:09:32 +00:00
|
|
|
func ReadConfigData(location string) ([]byte, error) {
|
|
|
|
if len(location) == 0 {
|
2014-11-20 10:00:36 +00:00
|
|
|
return nil, fmt.Errorf("location given but empty")
|
2014-11-10 17:09:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if location == "-" {
|
|
|
|
// Read from stdin.
|
|
|
|
data, err := ioutil.ReadAll(os.Stdin)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(data) == 0 {
|
|
|
|
return nil, fmt.Errorf(`Read from stdin specified ("-") but no data found`)
|
|
|
|
}
|
|
|
|
|
|
|
|
return data, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Use the location as a file path or URL.
|
|
|
|
return ReadConfigDataFromLocation(location)
|
|
|
|
}
|
|
|
|
|
2014-12-27 21:48:27 +00:00
|
|
|
// TODO: replace with resource.Builder
|
2014-11-10 17:09:32 +00:00
|
|
|
func ReadConfigDataFromLocation(location string) ([]byte, error) {
|
|
|
|
// we look for http:// or https:// to determine if valid URL, otherwise do normal file IO
|
|
|
|
if strings.Index(location, "http://") == 0 || strings.Index(location, "https://") == 0 {
|
|
|
|
resp, err := http.Get(location)
|
|
|
|
if err != nil {
|
2014-11-20 10:00:36 +00:00
|
|
|
return nil, fmt.Errorf("unable to access URL %s: %v\n", location, err)
|
2014-11-10 17:09:32 +00:00
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode != 200 {
|
2014-11-20 10:00:36 +00:00
|
|
|
return nil, fmt.Errorf("unable to read URL, server reported %d %s", resp.StatusCode, resp.Status)
|
2014-11-10 17:09:32 +00:00
|
|
|
}
|
|
|
|
data, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
2014-11-20 10:00:36 +00:00
|
|
|
return nil, fmt.Errorf("unable to read URL %s: %v\n", location, err)
|
2014-11-10 17:09:32 +00:00
|
|
|
}
|
|
|
|
return data, nil
|
|
|
|
} else {
|
|
|
|
data, err := ioutil.ReadFile(location)
|
|
|
|
if err != nil {
|
2014-11-20 10:00:36 +00:00
|
|
|
return nil, fmt.Errorf("unable to read %s: %v\n", location, err)
|
2014-11-10 17:09:32 +00:00
|
|
|
}
|
|
|
|
return data, nil
|
|
|
|
}
|
|
|
|
}
|
2015-01-15 03:29:13 +00:00
|
|
|
|
2015-01-31 18:59:08 +00:00
|
|
|
func Merge(dst runtime.Object, fragment, kind string) (runtime.Object, error) {
|
2015-01-15 03:29:13 +00:00
|
|
|
// Ok, this is a little hairy, we'd rather not force the user to specify a kind for their JSON
|
|
|
|
// So we pull it into a map, add the Kind field, and then reserialize.
|
|
|
|
// We also pull the apiVersion for proper parsing
|
|
|
|
var intermediate interface{}
|
|
|
|
if err := json.Unmarshal([]byte(fragment), &intermediate); err != nil {
|
2015-01-31 18:59:08 +00:00
|
|
|
return nil, err
|
2015-01-15 03:29:13 +00:00
|
|
|
}
|
|
|
|
dataMap, ok := intermediate.(map[string]interface{})
|
|
|
|
if !ok {
|
2015-01-31 18:59:08 +00:00
|
|
|
return nil, fmt.Errorf("Expected a map, found something else: %s", fragment)
|
2015-01-15 03:29:13 +00:00
|
|
|
}
|
|
|
|
version, found := dataMap["apiVersion"]
|
|
|
|
if !found {
|
2015-01-31 18:59:08 +00:00
|
|
|
return nil, fmt.Errorf("Inline JSON requires an apiVersion field")
|
2015-01-15 03:29:13 +00:00
|
|
|
}
|
|
|
|
versionString, ok := version.(string)
|
|
|
|
if !ok {
|
2015-01-31 18:59:08 +00:00
|
|
|
return nil, fmt.Errorf("apiVersion must be a string")
|
2015-01-15 03:29:13 +00:00
|
|
|
}
|
|
|
|
|
2015-01-31 18:59:08 +00:00
|
|
|
codec := runtime.CodecFor(api.Scheme, versionString)
|
|
|
|
// encode dst into versioned json and apply fragment directly too it
|
|
|
|
target, err := codec.Encode(dst)
|
2015-01-15 03:29:13 +00:00
|
|
|
if err != nil {
|
2015-01-31 18:59:08 +00:00
|
|
|
return nil, err
|
2015-01-15 03:29:13 +00:00
|
|
|
}
|
2015-02-23 19:47:06 +00:00
|
|
|
patched, err := jsonpatch.MergePatch(target, []byte(fragment))
|
2015-01-15 03:29:13 +00:00
|
|
|
if err != nil {
|
2015-01-31 18:59:08 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
out, err := codec.Decode(patched)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2015-01-15 03:29:13 +00:00
|
|
|
}
|
2015-01-31 18:59:08 +00:00
|
|
|
return out, nil
|
2015-01-15 03:29:13 +00:00
|
|
|
}
|