mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-31 13:50:01 +00:00 
			
		
		
		
	godep restore pushd $GOPATH/src/github.com/appc/spec git co master popd go get go4.org/errorutil rm -rf Godeps godep save ./... git add vendor git add -f $(git ls-files --other vendor/) git co -- Godeps/LICENSES Godeps/.license_file_state Godeps/OWNERS
		
			
				
	
	
		
			419 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			419 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package jmespath
 | |
| 
 | |
| import (
 | |
| 	"errors"
 | |
| 	"reflect"
 | |
| 	"unicode"
 | |
| 	"unicode/utf8"
 | |
| )
 | |
| 
 | |
| /* This is a tree based interpreter.  It walks the AST and directly
 | |
|    interprets the AST to search through a JSON document.
 | |
| */
 | |
| 
 | |
| type treeInterpreter struct {
 | |
| 	fCall *functionCaller
 | |
| }
 | |
| 
 | |
| func newInterpreter() *treeInterpreter {
 | |
| 	interpreter := treeInterpreter{}
 | |
| 	interpreter.fCall = newFunctionCaller()
 | |
| 	return &interpreter
 | |
| }
 | |
| 
 | |
| type expRef struct {
 | |
| 	ref ASTNode
 | |
| }
 | |
| 
 | |
| // Execute takes an ASTNode and input data and interprets the AST directly.
 | |
| // It will produce the result of applying the JMESPath expression associated
 | |
| // with the ASTNode to the input data "value".
 | |
| func (intr *treeInterpreter) Execute(node ASTNode, value interface{}) (interface{}, error) {
 | |
| 	switch node.nodeType {
 | |
| 	case ASTComparator:
 | |
| 		left, err := intr.Execute(node.children[0], value)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		right, err := intr.Execute(node.children[1], value)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		switch node.value {
 | |
| 		case tEQ:
 | |
| 			return objsEqual(left, right), nil
 | |
| 		case tNE:
 | |
| 			return !objsEqual(left, right), nil
 | |
| 		}
 | |
| 		leftNum, ok := left.(float64)
 | |
| 		if !ok {
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		rightNum, ok := right.(float64)
 | |
| 		if !ok {
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		switch node.value {
 | |
| 		case tGT:
 | |
| 			return leftNum > rightNum, nil
 | |
| 		case tGTE:
 | |
| 			return leftNum >= rightNum, nil
 | |
| 		case tLT:
 | |
| 			return leftNum < rightNum, nil
 | |
| 		case tLTE:
 | |
| 			return leftNum <= rightNum, nil
 | |
| 		}
 | |
| 	case ASTExpRef:
 | |
| 		return expRef{ref: node.children[0]}, nil
 | |
| 	case ASTFunctionExpression:
 | |
| 		resolvedArgs := []interface{}{}
 | |
| 		for _, arg := range node.children {
 | |
| 			current, err := intr.Execute(arg, value)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			resolvedArgs = append(resolvedArgs, current)
 | |
| 		}
 | |
| 		return intr.fCall.CallFunction(node.value.(string), resolvedArgs, intr)
 | |
| 	case ASTField:
 | |
| 		if m, ok := value.(map[string]interface{}); ok {
 | |
| 			key := node.value.(string)
 | |
| 			return m[key], nil
 | |
| 		}
 | |
| 		return intr.fieldFromStruct(node.value.(string), value)
 | |
| 	case ASTFilterProjection:
 | |
| 		left, err := intr.Execute(node.children[0], value)
 | |
| 		if err != nil {
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		sliceType, ok := left.([]interface{})
 | |
| 		if !ok {
 | |
| 			if isSliceType(left) {
 | |
| 				return intr.filterProjectionWithReflection(node, left)
 | |
| 			}
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		compareNode := node.children[2]
 | |
| 		collected := []interface{}{}
 | |
| 		for _, element := range sliceType {
 | |
| 			result, err := intr.Execute(compareNode, element)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			if !isFalse(result) {
 | |
| 				current, err := intr.Execute(node.children[1], element)
 | |
| 				if err != nil {
 | |
| 					return nil, err
 | |
| 				}
 | |
| 				if current != nil {
 | |
| 					collected = append(collected, current)
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		return collected, nil
 | |
| 	case ASTFlatten:
 | |
| 		left, err := intr.Execute(node.children[0], value)
 | |
| 		if err != nil {
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		sliceType, ok := left.([]interface{})
 | |
| 		if !ok {
 | |
| 			// If we can't type convert to []interface{}, there's
 | |
| 			// a chance this could still work via reflection if we're
 | |
| 			// dealing with user provided types.
 | |
| 			if isSliceType(left) {
 | |
| 				return intr.flattenWithReflection(left)
 | |
| 			}
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		flattened := []interface{}{}
 | |
| 		for _, element := range sliceType {
 | |
| 			if elementSlice, ok := element.([]interface{}); ok {
 | |
| 				flattened = append(flattened, elementSlice...)
 | |
| 			} else if isSliceType(element) {
 | |
| 				reflectFlat := []interface{}{}
 | |
| 				v := reflect.ValueOf(element)
 | |
| 				for i := 0; i < v.Len(); i++ {
 | |
| 					reflectFlat = append(reflectFlat, v.Index(i).Interface())
 | |
| 				}
 | |
| 				flattened = append(flattened, reflectFlat...)
 | |
| 			} else {
 | |
| 				flattened = append(flattened, element)
 | |
| 			}
 | |
| 		}
 | |
| 		return flattened, nil
 | |
| 	case ASTIdentity, ASTCurrentNode:
 | |
| 		return value, nil
 | |
| 	case ASTIndex:
 | |
| 		if sliceType, ok := value.([]interface{}); ok {
 | |
| 			index := node.value.(int)
 | |
| 			if index < 0 {
 | |
| 				index += len(sliceType)
 | |
| 			}
 | |
| 			if index < len(sliceType) && index >= 0 {
 | |
| 				return sliceType[index], nil
 | |
| 			}
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		// Otherwise try via reflection.
 | |
| 		rv := reflect.ValueOf(value)
 | |
| 		if rv.Kind() == reflect.Slice {
 | |
| 			index := node.value.(int)
 | |
| 			if index < 0 {
 | |
| 				index += rv.Len()
 | |
| 			}
 | |
| 			if index < rv.Len() && index >= 0 {
 | |
| 				v := rv.Index(index)
 | |
| 				return v.Interface(), nil
 | |
| 			}
 | |
| 		}
 | |
| 		return nil, nil
 | |
| 	case ASTKeyValPair:
 | |
| 		return intr.Execute(node.children[0], value)
 | |
| 	case ASTLiteral:
 | |
| 		return node.value, nil
 | |
| 	case ASTMultiSelectHash:
 | |
| 		if value == nil {
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		collected := make(map[string]interface{})
 | |
| 		for _, child := range node.children {
 | |
| 			current, err := intr.Execute(child, value)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			key := child.value.(string)
 | |
| 			collected[key] = current
 | |
| 		}
 | |
| 		return collected, nil
 | |
| 	case ASTMultiSelectList:
 | |
| 		if value == nil {
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		collected := []interface{}{}
 | |
| 		for _, child := range node.children {
 | |
| 			current, err := intr.Execute(child, value)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			collected = append(collected, current)
 | |
| 		}
 | |
| 		return collected, nil
 | |
| 	case ASTOrExpression:
 | |
| 		matched, err := intr.Execute(node.children[0], value)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		if isFalse(matched) {
 | |
| 			matched, err = intr.Execute(node.children[1], value)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 		}
 | |
| 		return matched, nil
 | |
| 	case ASTAndExpression:
 | |
| 		matched, err := intr.Execute(node.children[0], value)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		if isFalse(matched) {
 | |
| 			return matched, nil
 | |
| 		}
 | |
| 		return intr.Execute(node.children[1], value)
 | |
| 	case ASTNotExpression:
 | |
| 		matched, err := intr.Execute(node.children[0], value)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		if isFalse(matched) {
 | |
| 			return true, nil
 | |
| 		}
 | |
| 		return false, nil
 | |
| 	case ASTPipe:
 | |
| 		result := value
 | |
| 		var err error
 | |
| 		for _, child := range node.children {
 | |
| 			result, err = intr.Execute(child, result)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 		}
 | |
| 		return result, nil
 | |
| 	case ASTProjection:
 | |
| 		left, err := intr.Execute(node.children[0], value)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		sliceType, ok := left.([]interface{})
 | |
| 		if !ok {
 | |
| 			if isSliceType(left) {
 | |
| 				return intr.projectWithReflection(node, left)
 | |
| 			}
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		collected := []interface{}{}
 | |
| 		var current interface{}
 | |
| 		for _, element := range sliceType {
 | |
| 			current, err = intr.Execute(node.children[1], element)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			if current != nil {
 | |
| 				collected = append(collected, current)
 | |
| 			}
 | |
| 		}
 | |
| 		return collected, nil
 | |
| 	case ASTSubexpression, ASTIndexExpression:
 | |
| 		left, err := intr.Execute(node.children[0], value)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		return intr.Execute(node.children[1], left)
 | |
| 	case ASTSlice:
 | |
| 		sliceType, ok := value.([]interface{})
 | |
| 		if !ok {
 | |
| 			if isSliceType(value) {
 | |
| 				return intr.sliceWithReflection(node, value)
 | |
| 			}
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		parts := node.value.([]*int)
 | |
| 		sliceParams := make([]sliceParam, 3)
 | |
| 		for i, part := range parts {
 | |
| 			if part != nil {
 | |
| 				sliceParams[i].Specified = true
 | |
| 				sliceParams[i].N = *part
 | |
| 			}
 | |
| 		}
 | |
| 		return slice(sliceType, sliceParams)
 | |
| 	case ASTValueProjection:
 | |
| 		left, err := intr.Execute(node.children[0], value)
 | |
| 		if err != nil {
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		mapType, ok := left.(map[string]interface{})
 | |
| 		if !ok {
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		values := make([]interface{}, len(mapType))
 | |
| 		for _, value := range mapType {
 | |
| 			values = append(values, value)
 | |
| 		}
 | |
| 		collected := []interface{}{}
 | |
| 		for _, element := range values {
 | |
| 			current, err := intr.Execute(node.children[1], element)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			if current != nil {
 | |
| 				collected = append(collected, current)
 | |
| 			}
 | |
| 		}
 | |
| 		return collected, nil
 | |
| 	}
 | |
| 	return nil, errors.New("Unknown AST node: " + node.nodeType.String())
 | |
| }
 | |
| 
 | |
| func (intr *treeInterpreter) fieldFromStruct(key string, value interface{}) (interface{}, error) {
 | |
| 	rv := reflect.ValueOf(value)
 | |
| 	first, n := utf8.DecodeRuneInString(key)
 | |
| 	fieldName := string(unicode.ToUpper(first)) + key[n:]
 | |
| 	if rv.Kind() == reflect.Struct {
 | |
| 		v := rv.FieldByName(fieldName)
 | |
| 		if !v.IsValid() {
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		return v.Interface(), nil
 | |
| 	} else if rv.Kind() == reflect.Ptr {
 | |
| 		// Handle multiple levels of indirection?
 | |
| 		if rv.IsNil() {
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		rv = rv.Elem()
 | |
| 		v := rv.FieldByName(fieldName)
 | |
| 		if !v.IsValid() {
 | |
| 			return nil, nil
 | |
| 		}
 | |
| 		return v.Interface(), nil
 | |
| 	}
 | |
| 	return nil, nil
 | |
| }
 | |
| 
 | |
| func (intr *treeInterpreter) flattenWithReflection(value interface{}) (interface{}, error) {
 | |
| 	v := reflect.ValueOf(value)
 | |
| 	flattened := []interface{}{}
 | |
| 	for i := 0; i < v.Len(); i++ {
 | |
| 		element := v.Index(i).Interface()
 | |
| 		if reflect.TypeOf(element).Kind() == reflect.Slice {
 | |
| 			// Then insert the contents of the element
 | |
| 			// slice into the flattened slice,
 | |
| 			// i.e flattened = append(flattened, mySlice...)
 | |
| 			elementV := reflect.ValueOf(element)
 | |
| 			for j := 0; j < elementV.Len(); j++ {
 | |
| 				flattened = append(
 | |
| 					flattened, elementV.Index(j).Interface())
 | |
| 			}
 | |
| 		} else {
 | |
| 			flattened = append(flattened, element)
 | |
| 		}
 | |
| 	}
 | |
| 	return flattened, nil
 | |
| }
 | |
| 
 | |
| func (intr *treeInterpreter) sliceWithReflection(node ASTNode, value interface{}) (interface{}, error) {
 | |
| 	v := reflect.ValueOf(value)
 | |
| 	parts := node.value.([]*int)
 | |
| 	sliceParams := make([]sliceParam, 3)
 | |
| 	for i, part := range parts {
 | |
| 		if part != nil {
 | |
| 			sliceParams[i].Specified = true
 | |
| 			sliceParams[i].N = *part
 | |
| 		}
 | |
| 	}
 | |
| 	final := []interface{}{}
 | |
| 	for i := 0; i < v.Len(); i++ {
 | |
| 		element := v.Index(i).Interface()
 | |
| 		final = append(final, element)
 | |
| 	}
 | |
| 	return slice(final, sliceParams)
 | |
| }
 | |
| 
 | |
| func (intr *treeInterpreter) filterProjectionWithReflection(node ASTNode, value interface{}) (interface{}, error) {
 | |
| 	compareNode := node.children[2]
 | |
| 	collected := []interface{}{}
 | |
| 	v := reflect.ValueOf(value)
 | |
| 	for i := 0; i < v.Len(); i++ {
 | |
| 		element := v.Index(i).Interface()
 | |
| 		result, err := intr.Execute(compareNode, element)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		if !isFalse(result) {
 | |
| 			current, err := intr.Execute(node.children[1], element)
 | |
| 			if err != nil {
 | |
| 				return nil, err
 | |
| 			}
 | |
| 			if current != nil {
 | |
| 				collected = append(collected, current)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return collected, nil
 | |
| }
 | |
| 
 | |
| func (intr *treeInterpreter) projectWithReflection(node ASTNode, value interface{}) (interface{}, error) {
 | |
| 	collected := []interface{}{}
 | |
| 	v := reflect.ValueOf(value)
 | |
| 	for i := 0; i < v.Len(); i++ {
 | |
| 		element := v.Index(i).Interface()
 | |
| 		result, err := intr.Execute(node.children[1], element)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		if result != nil {
 | |
| 			collected = append(collected, result)
 | |
| 		}
 | |
| 	}
 | |
| 	return collected, nil
 | |
| }
 |