mirror of
				https://github.com/optim-enterprises-bv/kubernetes.git
				synced 2025-11-04 12:18:16 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			252 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			252 lines
		
	
	
		
			7.2 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
/*
 | 
						|
Copyright 2014 The Kubernetes Authors.
 | 
						|
 | 
						|
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 kubectl
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"io"
 | 
						|
	"strings"
 | 
						|
 | 
						|
	"github.com/emicklei/go-restful/swagger"
 | 
						|
 | 
						|
	"k8s.io/apimachinery/pkg/api/meta"
 | 
						|
	apiutil "k8s.io/kubernetes/pkg/api/util"
 | 
						|
)
 | 
						|
 | 
						|
var allModels = make(map[string]*swagger.NamedModel)
 | 
						|
var recursive = false // this is global for convenience, can become int for multiple levels
 | 
						|
 | 
						|
// SplitAndParseResourceRequest separates the users input into a model and fields
 | 
						|
func SplitAndParseResourceRequest(inResource string, mapper meta.RESTMapper) (string, []string, error) {
 | 
						|
	inResource, fieldsPath := splitDotNotation(inResource)
 | 
						|
	inResource, _ = mapper.ResourceSingularizer(inResource)
 | 
						|
	return inResource, fieldsPath, nil
 | 
						|
}
 | 
						|
 | 
						|
// PrintModelDescription prints the description of a specific model or dot path
 | 
						|
func PrintModelDescription(inModel string, fieldsPath []string, w io.Writer, swaggerSchema *swagger.ApiDeclaration, r bool) error {
 | 
						|
	recursive = r // this is global for convenience
 | 
						|
	apiVer := apiutil.GetVersion(swaggerSchema.ApiVersion) + "."
 | 
						|
 | 
						|
	var pointedModel *swagger.NamedModel
 | 
						|
	for i := range swaggerSchema.Models.List {
 | 
						|
		name := swaggerSchema.Models.List[i].Name
 | 
						|
 | 
						|
		allModels[name] = &swaggerSchema.Models.List[i]
 | 
						|
		if strings.ToLower(name) == strings.ToLower(apiVer+inModel) {
 | 
						|
			pointedModel = &swaggerSchema.Models.List[i]
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if pointedModel == nil {
 | 
						|
		return fmt.Errorf("requested resource %q is not defined", inModel)
 | 
						|
	}
 | 
						|
 | 
						|
	if len(fieldsPath) == 0 {
 | 
						|
		return printTopLevelResourceInfo(w, pointedModel)
 | 
						|
	}
 | 
						|
 | 
						|
	var pointedModelAsProp *swagger.NamedModelProperty
 | 
						|
	for _, field := range fieldsPath {
 | 
						|
		if prop, nextModel, isModel := getField(pointedModel, field); prop != nil {
 | 
						|
			if isModel {
 | 
						|
				pointedModelAsProp = prop
 | 
						|
				pointedModel = allModels[nextModel]
 | 
						|
			} else {
 | 
						|
				return printPrimitive(w, prop)
 | 
						|
			}
 | 
						|
		} else {
 | 
						|
			return fmt.Errorf("field %q does not exist", field)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return printModelInfo(w, pointedModel, pointedModelAsProp)
 | 
						|
}
 | 
						|
 | 
						|
func splitDotNotation(model string) (string, []string) {
 | 
						|
	var fieldsPath []string
 | 
						|
	dotModel := strings.Split(model, ".")
 | 
						|
	if len(dotModel) >= 1 {
 | 
						|
		fieldsPath = dotModel[1:]
 | 
						|
	}
 | 
						|
	return dotModel[0], fieldsPath
 | 
						|
}
 | 
						|
 | 
						|
func getPointedModel(prop *swagger.ModelProperty) (string, bool) {
 | 
						|
	if prop.Ref != nil {
 | 
						|
		return *prop.Ref, true
 | 
						|
	} else if *prop.Type == "array" && prop.Items.Ref != nil {
 | 
						|
		return *prop.Items.Ref, true
 | 
						|
	}
 | 
						|
	return "", false
 | 
						|
}
 | 
						|
 | 
						|
func getField(model *swagger.NamedModel, sField string) (*swagger.NamedModelProperty, string, bool) {
 | 
						|
	for _, prop := range model.Model.Properties.List {
 | 
						|
		if prop.Name == sField {
 | 
						|
			pointedModel, isModel := getPointedModel(&prop.Property)
 | 
						|
			return &prop, pointedModel, isModel
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return nil, "", false
 | 
						|
}
 | 
						|
 | 
						|
func printModelInfo(w io.Writer, model *swagger.NamedModel, modelProp *swagger.NamedModelProperty) error {
 | 
						|
	t, _ := getFieldType(&modelProp.Property)
 | 
						|
	fmt.Fprintf(w, "RESOURCE: %s <%s>\n\n", modelProp.Name, t)
 | 
						|
	fieldDesc, _ := wrapAndIndentText(modelProp.Property.Description, "    ", 80)
 | 
						|
	fmt.Fprintf(w, "DESCRIPTION:\n%s\n\n%s\n", fieldDesc, indentText(model.Model.Description, "    "))
 | 
						|
	return printFields(w, model)
 | 
						|
}
 | 
						|
 | 
						|
func printPrimitive(w io.Writer, field *swagger.NamedModelProperty) error {
 | 
						|
	t, _ := getFieldType(&field.Property)
 | 
						|
	fmt.Fprintf(w, "FIELD: %s <%s>\n\n", field.Name, t)
 | 
						|
	d, _ := wrapAndIndentText(field.Property.Description, "    ", 80)
 | 
						|
	fmt.Fprintf(w, "DESCRIPTION:\n%s\n", d)
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func printTopLevelResourceInfo(w io.Writer, model *swagger.NamedModel) error {
 | 
						|
	fmt.Fprintf(w, "DESCRIPTION:\n%s\n", model.Model.Description)
 | 
						|
	return printFields(w, model)
 | 
						|
}
 | 
						|
 | 
						|
func printFields(w io.Writer, model *swagger.NamedModel) error {
 | 
						|
	fmt.Fprint(w, "\nFIELDS:\n")
 | 
						|
	for _, field := range model.Model.Properties.List {
 | 
						|
		fieldType, err := getFieldType(&field.Property)
 | 
						|
		if err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
 | 
						|
		if arrayContains(model.Model.Required, field.Name) {
 | 
						|
			fmt.Fprintf(w, "   %s\t<%s> -required-\n", field.Name, fieldType)
 | 
						|
		} else {
 | 
						|
			fmt.Fprintf(w, "   %s\t<%s>\n", field.Name, fieldType)
 | 
						|
		}
 | 
						|
 | 
						|
		if recursive {
 | 
						|
			pointedModel, isModel := getPointedModel(&field.Property)
 | 
						|
			if isModel {
 | 
						|
				for _, nestedField := range allModels[pointedModel].Model.Properties.List {
 | 
						|
					t, _ := getFieldType(&nestedField.Property)
 | 
						|
					fmt.Fprintf(w, "       %s\t<%s>\n", nestedField.Name, t)
 | 
						|
				}
 | 
						|
			}
 | 
						|
		} else {
 | 
						|
			fieldDesc, _ := wrapAndIndentText(field.Property.Description, "    ", 80)
 | 
						|
			fmt.Fprintf(w, "%s\n\n", fieldDesc)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	fmt.Fprint(w, "\n")
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func getFieldType(prop *swagger.ModelProperty) (string, error) {
 | 
						|
	if prop.Type == nil {
 | 
						|
		return "Object", nil
 | 
						|
	} else if *prop.Type == "any" {
 | 
						|
		// Swagger Spec doesn't return information for maps.
 | 
						|
		return "map[string]string", nil
 | 
						|
	} else if *prop.Type == "array" {
 | 
						|
		if prop.Items == nil {
 | 
						|
			return "", fmt.Errorf("error in swagger spec. Property: %v contains an array without type", prop)
 | 
						|
		}
 | 
						|
		if prop.Items.Ref != nil {
 | 
						|
			fieldType := "[]Object"
 | 
						|
			return fieldType, nil
 | 
						|
		}
 | 
						|
		fieldType := "[]" + *prop.Items.Type
 | 
						|
		return fieldType, nil
 | 
						|
	}
 | 
						|
	return *prop.Type, nil
 | 
						|
}
 | 
						|
 | 
						|
func wrapAndIndentText(desc, indent string, lim int) (string, error) {
 | 
						|
	words := strings.Split(strings.Replace(strings.TrimSpace(desc), "\n", " ", -1), " ")
 | 
						|
	n := len(words)
 | 
						|
 | 
						|
	for i := 0; i < n; i++ {
 | 
						|
		if len(words[i]) > lim {
 | 
						|
			if strings.Contains(words[i], "/") {
 | 
						|
				s := breakURL(words[i])
 | 
						|
				words = append(words[:i], append(s, words[i+1:]...)...)
 | 
						|
				i = i + len(s) - 1
 | 
						|
			} else {
 | 
						|
				fmt.Println(len(words[i]))
 | 
						|
				return "", fmt.Errorf("there are words longer that the break limit is")
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	var lines []string
 | 
						|
	line := []string{indent}
 | 
						|
	lineL := len(indent)
 | 
						|
	for i := 0; i < len(words); i++ {
 | 
						|
		w := words[i]
 | 
						|
 | 
						|
		if strings.HasSuffix(w, "/") && lineL+len(w)-1 < lim {
 | 
						|
			prev := line[len(line)-1]
 | 
						|
			if strings.HasSuffix(prev, "/") {
 | 
						|
				if i+1 < len(words)-1 && !strings.HasSuffix(words[i+1], "/") {
 | 
						|
					w = strings.TrimSuffix(w, "/")
 | 
						|
				}
 | 
						|
 | 
						|
				line[len(line)-1] = prev + w
 | 
						|
				lineL += len(w)
 | 
						|
			} else {
 | 
						|
				line = append(line, w)
 | 
						|
				lineL += len(w) + 1
 | 
						|
			}
 | 
						|
		} else if lineL+len(w) < lim {
 | 
						|
			line = append(line, w)
 | 
						|
			lineL += len(w) + 1
 | 
						|
		} else {
 | 
						|
			lines = append(lines, strings.Join(line, " "))
 | 
						|
			line = []string{indent, w}
 | 
						|
			lineL = len(indent) + len(w)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	lines = append(lines, strings.Join(line, " "))
 | 
						|
 | 
						|
	return strings.Join(lines, "\n"), nil
 | 
						|
}
 | 
						|
 | 
						|
func breakURL(url string) []string {
 | 
						|
	var buf []string
 | 
						|
	for _, part := range strings.Split(url, "/") {
 | 
						|
		buf = append(buf, part+"/")
 | 
						|
	}
 | 
						|
	return buf
 | 
						|
}
 | 
						|
 | 
						|
func indentText(text, indent string) string {
 | 
						|
	lines := strings.Split(text, "\n")
 | 
						|
	for i := range lines {
 | 
						|
		lines[i] = indent + lines[i]
 | 
						|
	}
 | 
						|
	return strings.Join(lines, "\n")
 | 
						|
}
 | 
						|
 | 
						|
func arrayContains(s []string, e string) bool {
 | 
						|
	for _, a := range s {
 | 
						|
		if a == e {
 | 
						|
			return true
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return false
 | 
						|
}
 |