mirror of
				https://gitea.com/Lydanne/buildx.git
				synced 2025-11-04 01:53:42 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			262 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			262 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package hclparser
 | 
						|
 | 
						|
import (
 | 
						|
	"errors"
 | 
						|
	"path"
 | 
						|
	"strings"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"github.com/hashicorp/go-cty-funcs/cidr"
 | 
						|
	"github.com/hashicorp/go-cty-funcs/crypto"
 | 
						|
	"github.com/hashicorp/go-cty-funcs/encoding"
 | 
						|
	"github.com/hashicorp/go-cty-funcs/uuid"
 | 
						|
	"github.com/hashicorp/hcl/v2/ext/tryfunc"
 | 
						|
	"github.com/hashicorp/hcl/v2/ext/typeexpr"
 | 
						|
	"github.com/zclconf/go-cty/cty"
 | 
						|
	"github.com/zclconf/go-cty/cty/function"
 | 
						|
	"github.com/zclconf/go-cty/cty/function/stdlib"
 | 
						|
)
 | 
						|
 | 
						|
type funcDef struct {
 | 
						|
	name    string
 | 
						|
	fn      function.Function
 | 
						|
	factory func() function.Function
 | 
						|
}
 | 
						|
 | 
						|
var stdlibFunctions = []funcDef{
 | 
						|
	{name: "absolute", fn: stdlib.AbsoluteFunc},
 | 
						|
	{name: "add", fn: stdlib.AddFunc},
 | 
						|
	{name: "and", fn: stdlib.AndFunc},
 | 
						|
	{name: "base64decode", fn: encoding.Base64DecodeFunc},
 | 
						|
	{name: "base64encode", fn: encoding.Base64EncodeFunc},
 | 
						|
	{name: "basename", factory: basenameFunc},
 | 
						|
	{name: "bcrypt", fn: crypto.BcryptFunc},
 | 
						|
	{name: "byteslen", fn: stdlib.BytesLenFunc},
 | 
						|
	{name: "bytesslice", fn: stdlib.BytesSliceFunc},
 | 
						|
	{name: "can", fn: tryfunc.CanFunc},
 | 
						|
	{name: "ceil", fn: stdlib.CeilFunc},
 | 
						|
	{name: "chomp", fn: stdlib.ChompFunc},
 | 
						|
	{name: "chunklist", fn: stdlib.ChunklistFunc},
 | 
						|
	{name: "cidrhost", fn: cidr.HostFunc},
 | 
						|
	{name: "cidrnetmask", fn: cidr.NetmaskFunc},
 | 
						|
	{name: "cidrsubnet", fn: cidr.SubnetFunc},
 | 
						|
	{name: "cidrsubnets", fn: cidr.SubnetsFunc},
 | 
						|
	{name: "coalesce", fn: stdlib.CoalesceFunc},
 | 
						|
	{name: "coalescelist", fn: stdlib.CoalesceListFunc},
 | 
						|
	{name: "compact", fn: stdlib.CompactFunc},
 | 
						|
	{name: "concat", fn: stdlib.ConcatFunc},
 | 
						|
	{name: "contains", fn: stdlib.ContainsFunc},
 | 
						|
	{name: "convert", fn: typeexpr.ConvertFunc},
 | 
						|
	{name: "csvdecode", fn: stdlib.CSVDecodeFunc},
 | 
						|
	{name: "dirname", factory: dirnameFunc},
 | 
						|
	{name: "distinct", fn: stdlib.DistinctFunc},
 | 
						|
	{name: "divide", fn: stdlib.DivideFunc},
 | 
						|
	{name: "element", fn: stdlib.ElementFunc},
 | 
						|
	{name: "equal", fn: stdlib.EqualFunc},
 | 
						|
	{name: "flatten", fn: stdlib.FlattenFunc},
 | 
						|
	{name: "floor", fn: stdlib.FloorFunc},
 | 
						|
	{name: "format", fn: stdlib.FormatFunc},
 | 
						|
	{name: "formatdate", fn: stdlib.FormatDateFunc},
 | 
						|
	{name: "formatlist", fn: stdlib.FormatListFunc},
 | 
						|
	{name: "greaterthan", fn: stdlib.GreaterThanFunc},
 | 
						|
	{name: "greaterthanorequalto", fn: stdlib.GreaterThanOrEqualToFunc},
 | 
						|
	{name: "hasindex", fn: stdlib.HasIndexFunc},
 | 
						|
	{name: "indent", fn: stdlib.IndentFunc},
 | 
						|
	{name: "index", fn: stdlib.IndexFunc},
 | 
						|
	{name: "indexof", factory: indexOfFunc},
 | 
						|
	{name: "int", fn: stdlib.IntFunc},
 | 
						|
	{name: "join", fn: stdlib.JoinFunc},
 | 
						|
	{name: "jsondecode", fn: stdlib.JSONDecodeFunc},
 | 
						|
	{name: "jsonencode", fn: stdlib.JSONEncodeFunc},
 | 
						|
	{name: "keys", fn: stdlib.KeysFunc},
 | 
						|
	{name: "length", fn: stdlib.LengthFunc},
 | 
						|
	{name: "lessthan", fn: stdlib.LessThanFunc},
 | 
						|
	{name: "lessthanorequalto", fn: stdlib.LessThanOrEqualToFunc},
 | 
						|
	{name: "log", fn: stdlib.LogFunc},
 | 
						|
	{name: "lookup", fn: stdlib.LookupFunc},
 | 
						|
	{name: "lower", fn: stdlib.LowerFunc},
 | 
						|
	{name: "max", fn: stdlib.MaxFunc},
 | 
						|
	{name: "md5", fn: crypto.Md5Func},
 | 
						|
	{name: "merge", fn: stdlib.MergeFunc},
 | 
						|
	{name: "min", fn: stdlib.MinFunc},
 | 
						|
	{name: "modulo", fn: stdlib.ModuloFunc},
 | 
						|
	{name: "multiply", fn: stdlib.MultiplyFunc},
 | 
						|
	{name: "negate", fn: stdlib.NegateFunc},
 | 
						|
	{name: "not", fn: stdlib.NotFunc},
 | 
						|
	{name: "notequal", fn: stdlib.NotEqualFunc},
 | 
						|
	{name: "or", fn: stdlib.OrFunc},
 | 
						|
	{name: "parseint", fn: stdlib.ParseIntFunc},
 | 
						|
	{name: "pow", fn: stdlib.PowFunc},
 | 
						|
	{name: "range", fn: stdlib.RangeFunc},
 | 
						|
	{name: "regex_replace", fn: stdlib.RegexReplaceFunc},
 | 
						|
	{name: "regex", fn: stdlib.RegexFunc},
 | 
						|
	{name: "regexall", fn: stdlib.RegexAllFunc},
 | 
						|
	{name: "replace", fn: stdlib.ReplaceFunc},
 | 
						|
	{name: "reverse", fn: stdlib.ReverseFunc},
 | 
						|
	{name: "reverselist", fn: stdlib.ReverseListFunc},
 | 
						|
	{name: "rsadecrypt", fn: crypto.RsaDecryptFunc},
 | 
						|
	{name: "sanitize", factory: sanitizeFunc},
 | 
						|
	{name: "sethaselement", fn: stdlib.SetHasElementFunc},
 | 
						|
	{name: "setintersection", fn: stdlib.SetIntersectionFunc},
 | 
						|
	{name: "setproduct", fn: stdlib.SetProductFunc},
 | 
						|
	{name: "setsubtract", fn: stdlib.SetSubtractFunc},
 | 
						|
	{name: "setsymmetricdifference", fn: stdlib.SetSymmetricDifferenceFunc},
 | 
						|
	{name: "setunion", fn: stdlib.SetUnionFunc},
 | 
						|
	{name: "sha1", fn: crypto.Sha1Func},
 | 
						|
	{name: "sha256", fn: crypto.Sha256Func},
 | 
						|
	{name: "sha512", fn: crypto.Sha512Func},
 | 
						|
	{name: "signum", fn: stdlib.SignumFunc},
 | 
						|
	{name: "slice", fn: stdlib.SliceFunc},
 | 
						|
	{name: "sort", fn: stdlib.SortFunc},
 | 
						|
	{name: "split", fn: stdlib.SplitFunc},
 | 
						|
	{name: "strlen", fn: stdlib.StrlenFunc},
 | 
						|
	{name: "substr", fn: stdlib.SubstrFunc},
 | 
						|
	{name: "subtract", fn: stdlib.SubtractFunc},
 | 
						|
	{name: "timeadd", fn: stdlib.TimeAddFunc},
 | 
						|
	{name: "timestamp", factory: timestampFunc},
 | 
						|
	{name: "title", fn: stdlib.TitleFunc},
 | 
						|
	{name: "trim", fn: stdlib.TrimFunc},
 | 
						|
	{name: "trimprefix", fn: stdlib.TrimPrefixFunc},
 | 
						|
	{name: "trimspace", fn: stdlib.TrimSpaceFunc},
 | 
						|
	{name: "trimsuffix", fn: stdlib.TrimSuffixFunc},
 | 
						|
	{name: "try", fn: tryfunc.TryFunc},
 | 
						|
	{name: "upper", fn: stdlib.UpperFunc},
 | 
						|
	{name: "urlencode", fn: encoding.URLEncodeFunc},
 | 
						|
	{name: "uuidv4", fn: uuid.V4Func},
 | 
						|
	{name: "uuidv5", fn: uuid.V5Func},
 | 
						|
	{name: "values", fn: stdlib.ValuesFunc},
 | 
						|
	{name: "zipmap", fn: stdlib.ZipmapFunc},
 | 
						|
}
 | 
						|
 | 
						|
// indexOfFunc constructs a function that finds the element index for a given
 | 
						|
// value in a list.
 | 
						|
func indexOfFunc() function.Function {
 | 
						|
	return function.New(&function.Spec{
 | 
						|
		Params: []function.Parameter{
 | 
						|
			{
 | 
						|
				Name: "list",
 | 
						|
				Type: cty.DynamicPseudoType,
 | 
						|
			},
 | 
						|
			{
 | 
						|
				Name: "value",
 | 
						|
				Type: cty.DynamicPseudoType,
 | 
						|
			},
 | 
						|
		},
 | 
						|
		Type: function.StaticReturnType(cty.Number),
 | 
						|
		Impl: func(args []cty.Value, retType cty.Type) (ret cty.Value, err error) {
 | 
						|
			if !(args[0].Type().IsListType() || args[0].Type().IsTupleType()) {
 | 
						|
				return cty.NilVal, errors.New("argument must be a list or tuple")
 | 
						|
			}
 | 
						|
 | 
						|
			if !args[0].IsKnown() {
 | 
						|
				return cty.UnknownVal(cty.Number), nil
 | 
						|
			}
 | 
						|
 | 
						|
			if args[0].LengthInt() == 0 { // Easy path
 | 
						|
				return cty.NilVal, errors.New("cannot search an empty list")
 | 
						|
			}
 | 
						|
 | 
						|
			for it := args[0].ElementIterator(); it.Next(); {
 | 
						|
				i, v := it.Element()
 | 
						|
				eq, err := stdlib.Equal(v, args[1])
 | 
						|
				if err != nil {
 | 
						|
					return cty.NilVal, err
 | 
						|
				}
 | 
						|
				if !eq.IsKnown() {
 | 
						|
					return cty.UnknownVal(cty.Number), nil
 | 
						|
				}
 | 
						|
				if eq.True() {
 | 
						|
					return i, nil
 | 
						|
				}
 | 
						|
			}
 | 
						|
			return cty.NilVal, errors.New("item not found")
 | 
						|
		},
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
// basenameFunc constructs a function that returns the last element of a path.
 | 
						|
func basenameFunc() function.Function {
 | 
						|
	return function.New(&function.Spec{
 | 
						|
		Params: []function.Parameter{
 | 
						|
			{
 | 
						|
				Name: "path",
 | 
						|
				Type: cty.String,
 | 
						|
			},
 | 
						|
		},
 | 
						|
		Type: function.StaticReturnType(cty.String),
 | 
						|
		Impl: func(args []cty.Value, retType cty.Type) (cty.Value, error) {
 | 
						|
			in := args[0].AsString()
 | 
						|
			return cty.StringVal(path.Base(in)), nil
 | 
						|
		},
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
// dirnameFunc constructs a function that returns the directory of a path.
 | 
						|
func dirnameFunc() function.Function {
 | 
						|
	return function.New(&function.Spec{
 | 
						|
		Params: []function.Parameter{
 | 
						|
			{
 | 
						|
				Name: "path",
 | 
						|
				Type: cty.String,
 | 
						|
			},
 | 
						|
		},
 | 
						|
		Type: function.StaticReturnType(cty.String),
 | 
						|
		Impl: func(args []cty.Value, retType cty.Type) (cty.Value, error) {
 | 
						|
			in := args[0].AsString()
 | 
						|
			return cty.StringVal(path.Dir(in)), nil
 | 
						|
		},
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
// sanitizyFunc constructs a function that replaces all non-alphanumeric characters with a underscore,
 | 
						|
// leaving only characters that are valid for a Bake target name.
 | 
						|
func sanitizeFunc() function.Function {
 | 
						|
	return function.New(&function.Spec{
 | 
						|
		Params: []function.Parameter{
 | 
						|
			{
 | 
						|
				Name: "name",
 | 
						|
				Type: cty.String,
 | 
						|
			},
 | 
						|
		},
 | 
						|
		Type: function.StaticReturnType(cty.String),
 | 
						|
		Impl: func(args []cty.Value, retType cty.Type) (cty.Value, error) {
 | 
						|
			in := args[0].AsString()
 | 
						|
			// only [a-zA-Z0-9_-]+ is allowed
 | 
						|
			var b strings.Builder
 | 
						|
			for _, r := range in {
 | 
						|
				if r >= 'a' && r <= 'z' || r >= 'A' && r <= 'Z' || r >= '0' && r <= '9' || r == '_' || r == '-' {
 | 
						|
					b.WriteRune(r)
 | 
						|
				} else {
 | 
						|
					b.WriteRune('_')
 | 
						|
				}
 | 
						|
			}
 | 
						|
			return cty.StringVal(b.String()), nil
 | 
						|
		},
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
// timestampFunc constructs a function that returns a string representation of the current date and time.
 | 
						|
//
 | 
						|
// This function was imported from terraform's datetime utilities.
 | 
						|
func timestampFunc() function.Function {
 | 
						|
	return function.New(&function.Spec{
 | 
						|
		Params: []function.Parameter{},
 | 
						|
		Type:   function.StaticReturnType(cty.String),
 | 
						|
		Impl: func(args []cty.Value, retType cty.Type) (cty.Value, error) {
 | 
						|
			return cty.StringVal(time.Now().UTC().Format(time.RFC3339)), nil
 | 
						|
		},
 | 
						|
	})
 | 
						|
}
 | 
						|
 | 
						|
func Stdlib() map[string]function.Function {
 | 
						|
	funcs := make(map[string]function.Function, len(stdlibFunctions))
 | 
						|
	for _, v := range stdlibFunctions {
 | 
						|
		if v.factory != nil {
 | 
						|
			funcs[v.name] = v.factory()
 | 
						|
		} else {
 | 
						|
			funcs[v.name] = v.fn
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return funcs
 | 
						|
}
 |