mirror of
				https://gitea.com/Lydanne/buildx.git
				synced 2025-10-31 16:13:45 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			147 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			147 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package stdlib
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"strings"
 | |
| 	"unicode/utf8"
 | |
| 
 | |
| 	"github.com/zclconf/go-cty/cty"
 | |
| 	"github.com/zclconf/go-cty/cty/function"
 | |
| 	"github.com/zclconf/go-cty/cty/json"
 | |
| )
 | |
| 
 | |
| var JSONEncodeFunc = function.New(&function.Spec{
 | |
| 	Description: `Returns a string containing a JSON representation of the given value.`,
 | |
| 	Params: []function.Parameter{
 | |
| 		{
 | |
| 			Name:             "val",
 | |
| 			Type:             cty.DynamicPseudoType,
 | |
| 			AllowUnknown:     true,
 | |
| 			AllowDynamicType: true,
 | |
| 			AllowNull:        true,
 | |
| 		},
 | |
| 	},
 | |
| 	Type:         function.StaticReturnType(cty.String),
 | |
| 	RefineResult: refineNonNull,
 | |
| 	Impl: func(args []cty.Value, retType cty.Type) (cty.Value, error) {
 | |
| 		val := args[0]
 | |
| 		if !val.IsWhollyKnown() {
 | |
| 			// We can't serialize unknowns, so if the value is unknown or
 | |
| 			// contains any _nested_ unknowns then our result must be
 | |
| 			// unknown. However, we might still be able to at least constrain
 | |
| 			// the prefix of our string so that downstreams can sniff for
 | |
| 			// whether it's valid JSON and what result types it could have.
 | |
| 
 | |
| 			valRng := val.Range()
 | |
| 			if valRng.CouldBeNull() {
 | |
| 				// If null is possible then we can't constrain the result
 | |
| 				// beyond the type constraint, because the very first character
 | |
| 				// of the string is what distinguishes a null.
 | |
| 				return cty.UnknownVal(retType), nil
 | |
| 			}
 | |
| 			b := cty.UnknownVal(retType).Refine()
 | |
| 			ty := valRng.TypeConstraint()
 | |
| 			switch {
 | |
| 			case ty == cty.String:
 | |
| 				b = b.StringPrefixFull(`"`)
 | |
| 			case ty.IsObjectType() || ty.IsMapType():
 | |
| 				b = b.StringPrefixFull("{")
 | |
| 			case ty.IsTupleType() || ty.IsListType() || ty.IsSetType():
 | |
| 				b = b.StringPrefixFull("[")
 | |
| 			}
 | |
| 			return b.NewValue(), nil
 | |
| 		}
 | |
| 
 | |
| 		if val.IsNull() {
 | |
| 			return cty.StringVal("null"), nil
 | |
| 		}
 | |
| 
 | |
| 		buf, err := json.Marshal(val, val.Type())
 | |
| 		if err != nil {
 | |
| 			return cty.NilVal, err
 | |
| 		}
 | |
| 
 | |
| 		// json.Marshal should already produce a trimmed string, but we'll
 | |
| 		// make sure it always is because our unknown value refinements above
 | |
| 		// assume there will be no leading whitespace before the value.
 | |
| 		buf = bytes.TrimSpace(buf)
 | |
| 
 | |
| 		return cty.StringVal(string(buf)), nil
 | |
| 	},
 | |
| })
 | |
| 
 | |
| var JSONDecodeFunc = function.New(&function.Spec{
 | |
| 	Description: `Parses the given string as JSON and returns a value corresponding to what the JSON document describes.`,
 | |
| 	Params: []function.Parameter{
 | |
| 		{
 | |
| 			Name: "str",
 | |
| 			Type: cty.String,
 | |
| 		},
 | |
| 	},
 | |
| 	Type: func(args []cty.Value) (cty.Type, error) {
 | |
| 		str := args[0]
 | |
| 		if !str.IsKnown() {
 | |
| 			// If the string isn't known then we can't fully parse it, but
 | |
| 			// if the value has been refined with a prefix then we may at
 | |
| 			// least be able to reject obviously-invalid syntax and maybe
 | |
| 			// even predict the result type. It's safe to return a specific
 | |
| 			// result type only if parsing a full document with this prefix
 | |
| 			// would return exactly that type or fail with a syntax error.
 | |
| 			rng := str.Range()
 | |
| 			if prefix := strings.TrimSpace(rng.StringPrefix()); prefix != "" {
 | |
| 				// If we know at least one character then it should be one
 | |
| 				// of the few characters that can introduce a JSON value.
 | |
| 				switch r, _ := utf8.DecodeRuneInString(prefix); r {
 | |
| 				case '{', '[':
 | |
| 					// These can start object values and array values
 | |
| 					// respectively, but we can't actually form a full
 | |
| 					// object type constraint or tuple type constraint
 | |
| 					// without knowing all of the attributes, so we
 | |
| 					// will still return DynamicPseudoType in this case.
 | |
| 				case '"':
 | |
| 					// This means that the result will either be a string
 | |
| 					// or parsing will fail.
 | |
| 					return cty.String, nil
 | |
| 				case 't', 'f':
 | |
| 					// Must either be a boolean value or a syntax error.
 | |
| 					return cty.Bool, nil
 | |
| 				case '-', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '.':
 | |
| 					// These characters would all start the "number" production.
 | |
| 					return cty.Number, nil
 | |
| 				case 'n':
 | |
| 					// n is valid to begin the keyword "null" but that doesn't
 | |
| 					// give us any extra type information.
 | |
| 				default:
 | |
| 					// No other characters are valid as the beginning of a
 | |
| 					// JSON value, so we can safely return an early error.
 | |
| 					return cty.NilType, function.NewArgErrorf(0, "a JSON document cannot begin with the character %q", r)
 | |
| 				}
 | |
| 			}
 | |
| 			return cty.DynamicPseudoType, nil
 | |
| 		}
 | |
| 
 | |
| 		buf := []byte(str.AsString())
 | |
| 		return json.ImpliedType(buf)
 | |
| 	},
 | |
| 	Impl: func(args []cty.Value, retType cty.Type) (cty.Value, error) {
 | |
| 		buf := []byte(args[0].AsString())
 | |
| 		return json.Unmarshal(buf, retType)
 | |
| 	},
 | |
| })
 | |
| 
 | |
| // JSONEncode returns a JSON serialization of the given value.
 | |
| func JSONEncode(val cty.Value) (cty.Value, error) {
 | |
| 	return JSONEncodeFunc.Call([]cty.Value{val})
 | |
| }
 | |
| 
 | |
| // JSONDecode parses the given JSON string and, if it is valid, returns the
 | |
| // value it represents.
 | |
| //
 | |
| // Note that applying JSONDecode to the result of JSONEncode may not produce
 | |
| // an identically-typed result, since JSON encoding is lossy for cty Types.
 | |
| // The resulting value will consist only of primitive types, object types, and
 | |
| // tuple types.
 | |
| func JSONDecode(str cty.Value) (cty.Value, error) {
 | |
| 	return JSONDecodeFunc.Call([]cty.Value{str})
 | |
| }
 | 
