mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-31 05:40:42 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			119 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			119 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| /*
 | |
| Copyright 2018 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 fieldpath
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"strings"
 | |
| 
 | |
| 	"sigs.k8s.io/structured-merge-diff/v4/value"
 | |
| )
 | |
| 
 | |
| // Path describes how to select a potentially deeply-nested child field given a
 | |
| // containing object.
 | |
| type Path []PathElement
 | |
| 
 | |
| func (fp Path) String() string {
 | |
| 	strs := make([]string, len(fp))
 | |
| 	for i := range fp {
 | |
| 		strs[i] = fp[i].String()
 | |
| 	}
 | |
| 	return strings.Join(strs, "")
 | |
| }
 | |
| 
 | |
| // Equals returns true if the two paths are equivalent.
 | |
| func (fp Path) Equals(fp2 Path) bool {
 | |
| 	if len(fp) != len(fp2) {
 | |
| 		return false
 | |
| 	}
 | |
| 	for i := range fp {
 | |
| 		if !fp[i].Equals(fp2[i]) {
 | |
| 			return false
 | |
| 		}
 | |
| 	}
 | |
| 	return true
 | |
| }
 | |
| 
 | |
| // Less provides a lexical order for Paths.
 | |
| func (fp Path) Compare(rhs Path) int {
 | |
| 	i := 0
 | |
| 	for {
 | |
| 		if i >= len(fp) && i >= len(rhs) {
 | |
| 			// Paths are the same length and all items are equal.
 | |
| 			return 0
 | |
| 		}
 | |
| 		if i >= len(fp) {
 | |
| 			// LHS is shorter.
 | |
| 			return -1
 | |
| 		}
 | |
| 		if i >= len(rhs) {
 | |
| 			// RHS is shorter.
 | |
| 			return 1
 | |
| 		}
 | |
| 		if c := fp[i].Compare(rhs[i]); c != 0 {
 | |
| 			return c
 | |
| 		}
 | |
| 		// The items are equal; continue.
 | |
| 		i++
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (fp Path) Copy() Path {
 | |
| 	new := make(Path, len(fp))
 | |
| 	copy(new, fp)
 | |
| 	return new
 | |
| }
 | |
| 
 | |
| // MakePath constructs a Path. The parts may be PathElements, ints, strings.
 | |
| func MakePath(parts ...interface{}) (Path, error) {
 | |
| 	var fp Path
 | |
| 	for _, p := range parts {
 | |
| 		switch t := p.(type) {
 | |
| 		case PathElement:
 | |
| 			fp = append(fp, t)
 | |
| 		case int:
 | |
| 			// TODO: Understand schema and object and convert this to the
 | |
| 			// FieldSpecifier below if appropriate.
 | |
| 			fp = append(fp, PathElement{Index: &t})
 | |
| 		case string:
 | |
| 			fp = append(fp, PathElement{FieldName: &t})
 | |
| 		case *value.FieldList:
 | |
| 			if len(*t) == 0 {
 | |
| 				return nil, fmt.Errorf("associative list key type path elements must have at least one key (got zero)")
 | |
| 			}
 | |
| 			fp = append(fp, PathElement{Key: t})
 | |
| 		case value.Value:
 | |
| 			// TODO: understand schema and verify that this is a set type
 | |
| 			// TODO: make a copy of t
 | |
| 			fp = append(fp, PathElement{Value: &t})
 | |
| 		default:
 | |
| 			return nil, fmt.Errorf("unable to make %#v into a path element", p)
 | |
| 		}
 | |
| 	}
 | |
| 	return fp, nil
 | |
| }
 | |
| 
 | |
| // MakePathOrDie panics if parts can't be turned into a path. Good for things
 | |
| // that are known at complie time.
 | |
| func MakePathOrDie(parts ...interface{}) Path {
 | |
| 	fp, err := MakePath(parts...)
 | |
| 	if err != nil {
 | |
| 		panic(err)
 | |
| 	}
 | |
| 	return fp
 | |
| }
 |