mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-26 11:07:45 +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/kubernetes/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
 | |
| }
 |