mirror of https://github.com/docker/buildx.git
				
				
				
			
		
			
				
	
	
		
			474 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			474 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Go
		
	
	
	
| /*
 | |
|    Copyright 2020 The Compose Specification 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 template
 | |
| 
 | |
| import (
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"regexp"
 | |
| 	"sort"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/sirupsen/logrus"
 | |
| )
 | |
| 
 | |
| var delimiter = "\\$"
 | |
| var substitutionNamed = "[_a-z][_a-z0-9]*"
 | |
| var substitutionBraced = "[_a-z][_a-z0-9]*(?::?[-+?](.*))?"
 | |
| 
 | |
| var groupEscaped = "escaped"
 | |
| var groupNamed = "named"
 | |
| var groupBraced = "braced"
 | |
| var groupInvalid = "invalid"
 | |
| 
 | |
| var patternString = fmt.Sprintf(
 | |
| 	"%s(?i:(?P<%s>%s)|(?P<%s>%s)|{(?:(?P<%s>%s)}|(?P<%s>)))",
 | |
| 	delimiter,
 | |
| 	groupEscaped, delimiter,
 | |
| 	groupNamed, substitutionNamed,
 | |
| 	groupBraced, substitutionBraced,
 | |
| 	groupInvalid,
 | |
| )
 | |
| 
 | |
| var DefaultPattern = regexp.MustCompile(patternString)
 | |
| 
 | |
| // InvalidTemplateError is returned when a variable template is not in a valid
 | |
| // format
 | |
| type InvalidTemplateError struct {
 | |
| 	Template string
 | |
| }
 | |
| 
 | |
| func (e InvalidTemplateError) Error() string {
 | |
| 	return fmt.Sprintf("Invalid template: %#v", e.Template)
 | |
| }
 | |
| 
 | |
| // MissingRequiredError is returned when a variable template is missing
 | |
| type MissingRequiredError struct {
 | |
| 	Variable string
 | |
| 	Reason   string
 | |
| }
 | |
| 
 | |
| func (e MissingRequiredError) Error() string {
 | |
| 	if e.Reason != "" {
 | |
| 		return fmt.Sprintf("required variable %s is missing a value: %s", e.Variable, e.Reason)
 | |
| 	}
 | |
| 	return fmt.Sprintf("required variable %s is missing a value", e.Variable)
 | |
| }
 | |
| 
 | |
| // Mapping is a user-supplied function which maps from variable names to values.
 | |
| // Returns the value as a string and a bool indicating whether
 | |
| // the value is present, to distinguish between an empty string
 | |
| // and the absence of a value.
 | |
| type Mapping func(string) (string, bool)
 | |
| 
 | |
| // SubstituteFunc is a user-supplied function that apply substitution.
 | |
| // Returns the value as a string, a bool indicating if the function could apply
 | |
| // the substitution and an error.
 | |
| type SubstituteFunc func(string, Mapping) (string, bool, error)
 | |
| 
 | |
| // ReplacementFunc is a user-supplied function that is apply to the matching
 | |
| // substring. Returns the value as a string and an error.
 | |
| type ReplacementFunc func(string, Mapping, *Config) (string, error)
 | |
| 
 | |
| type Config struct {
 | |
| 	pattern         *regexp.Regexp
 | |
| 	substituteFunc  SubstituteFunc
 | |
| 	replacementFunc ReplacementFunc
 | |
| 	logging         bool
 | |
| }
 | |
| 
 | |
| type Option func(*Config)
 | |
| 
 | |
| func WithPattern(pattern *regexp.Regexp) Option {
 | |
| 	return func(cfg *Config) {
 | |
| 		cfg.pattern = pattern
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func WithSubstitutionFunction(subsFunc SubstituteFunc) Option {
 | |
| 	return func(cfg *Config) {
 | |
| 		cfg.substituteFunc = subsFunc
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func WithReplacementFunction(replacementFunc ReplacementFunc) Option {
 | |
| 	return func(cfg *Config) {
 | |
| 		cfg.replacementFunc = replacementFunc
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func WithoutLogging(cfg *Config) {
 | |
| 	cfg.logging = false
 | |
| }
 | |
| 
 | |
| // SubstituteWithOptions substitute variables in the string with their values.
 | |
| // It accepts additional options such as a custom function or pattern.
 | |
| func SubstituteWithOptions(template string, mapping Mapping, options ...Option) (string, error) {
 | |
| 	var returnErr error
 | |
| 
 | |
| 	cfg := &Config{
 | |
| 		pattern:         DefaultPattern,
 | |
| 		replacementFunc: DefaultReplacementFunc,
 | |
| 		logging:         true,
 | |
| 	}
 | |
| 	for _, o := range options {
 | |
| 		o(cfg)
 | |
| 	}
 | |
| 
 | |
| 	result := cfg.pattern.ReplaceAllStringFunc(template, func(substring string) string {
 | |
| 		replacement, err := cfg.replacementFunc(substring, mapping, cfg)
 | |
| 		if err != nil {
 | |
| 			// Add the template for template errors
 | |
| 			var tmplErr *InvalidTemplateError
 | |
| 			if errors.As(err, &tmplErr) {
 | |
| 				if tmplErr.Template == "" {
 | |
| 					tmplErr.Template = template
 | |
| 				}
 | |
| 			}
 | |
| 			// Save the first error to be returned
 | |
| 			if returnErr == nil {
 | |
| 				returnErr = err
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 		return replacement
 | |
| 	})
 | |
| 
 | |
| 	return result, returnErr
 | |
| }
 | |
| 
 | |
| func DefaultReplacementFunc(substring string, mapping Mapping, cfg *Config) (string, error) {
 | |
| 	value, _, err := DefaultReplacementAppliedFunc(substring, mapping, cfg)
 | |
| 	return value, err
 | |
| }
 | |
| 
 | |
| func DefaultReplacementAppliedFunc(substring string, mapping Mapping, cfg *Config) (string, bool, error) {
 | |
| 	pattern := cfg.pattern
 | |
| 	subsFunc := cfg.substituteFunc
 | |
| 	if subsFunc == nil {
 | |
| 		_, subsFunc = getSubstitutionFunctionForTemplate(substring)
 | |
| 	}
 | |
| 
 | |
| 	closingBraceIndex := getFirstBraceClosingIndex(substring)
 | |
| 	rest := ""
 | |
| 	if closingBraceIndex > -1 {
 | |
| 		rest = substring[closingBraceIndex+1:]
 | |
| 		substring = substring[0 : closingBraceIndex+1]
 | |
| 	}
 | |
| 
 | |
| 	matches := pattern.FindStringSubmatch(substring)
 | |
| 	groups := matchGroups(matches, pattern)
 | |
| 	if escaped := groups[groupEscaped]; escaped != "" {
 | |
| 		return escaped, true, nil
 | |
| 	}
 | |
| 
 | |
| 	braced := false
 | |
| 	substitution := groups[groupNamed]
 | |
| 	if substitution == "" {
 | |
| 		substitution = groups[groupBraced]
 | |
| 		braced = true
 | |
| 	}
 | |
| 
 | |
| 	if substitution == "" {
 | |
| 		return "", false, &InvalidTemplateError{}
 | |
| 	}
 | |
| 
 | |
| 	if braced {
 | |
| 		value, applied, err := subsFunc(substitution, mapping)
 | |
| 		if err != nil {
 | |
| 			return "", false, err
 | |
| 		}
 | |
| 		if applied {
 | |
| 			interpolatedNested, err := SubstituteWith(rest, mapping, pattern)
 | |
| 			if err != nil {
 | |
| 				return "", false, err
 | |
| 			}
 | |
| 			return value + interpolatedNested, true, nil
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	value, ok := mapping(substitution)
 | |
| 	if !ok && cfg.logging {
 | |
| 		logrus.Warnf("The %q variable is not set. Defaulting to a blank string.", substitution)
 | |
| 	}
 | |
| 
 | |
| 	return value, ok, nil
 | |
| }
 | |
| 
 | |
| // SubstituteWith substitute variables in the string with their values.
 | |
| // It accepts additional substitute function.
 | |
| func SubstituteWith(template string, mapping Mapping, pattern *regexp.Regexp, subsFuncs ...SubstituteFunc) (string, error) {
 | |
| 	options := []Option{
 | |
| 		WithPattern(pattern),
 | |
| 	}
 | |
| 	if len(subsFuncs) > 0 {
 | |
| 		options = append(options, WithSubstitutionFunction(subsFuncs[0]))
 | |
| 	}
 | |
| 
 | |
| 	return SubstituteWithOptions(template, mapping, options...)
 | |
| }
 | |
| 
 | |
| func getSubstitutionFunctionForTemplate(template string) (string, SubstituteFunc) {
 | |
| 	interpolationMapping := []struct {
 | |
| 		string
 | |
| 		SubstituteFunc
 | |
| 	}{
 | |
| 		{":?", requiredErrorWhenEmptyOrUnset},
 | |
| 		{"?", requiredErrorWhenUnset},
 | |
| 		{":-", defaultWhenEmptyOrUnset},
 | |
| 		{"-", defaultWhenUnset},
 | |
| 		{":+", defaultWhenNotEmpty},
 | |
| 		{"+", defaultWhenSet},
 | |
| 	}
 | |
| 	sort.Slice(interpolationMapping, func(i, j int) bool {
 | |
| 		idxI := strings.Index(template, interpolationMapping[i].string)
 | |
| 		idxJ := strings.Index(template, interpolationMapping[j].string)
 | |
| 		if idxI < 0 {
 | |
| 			return false
 | |
| 		}
 | |
| 		if idxJ < 0 {
 | |
| 			return true
 | |
| 		}
 | |
| 		return idxI < idxJ
 | |
| 	})
 | |
| 
 | |
| 	return interpolationMapping[0].string, interpolationMapping[0].SubstituteFunc
 | |
| }
 | |
| 
 | |
| func getFirstBraceClosingIndex(s string) int {
 | |
| 	openVariableBraces := 0
 | |
| 	for i := 0; i < len(s); i++ {
 | |
| 		if s[i] == '}' {
 | |
| 			openVariableBraces--
 | |
| 			if openVariableBraces == 0 {
 | |
| 				return i
 | |
| 			}
 | |
| 		}
 | |
| 		if s[i] == '{' {
 | |
| 			openVariableBraces++
 | |
| 			i++
 | |
| 		}
 | |
| 	}
 | |
| 	return -1
 | |
| }
 | |
| 
 | |
| // Substitute variables in the string with their values
 | |
| func Substitute(template string, mapping Mapping) (string, error) {
 | |
| 	return SubstituteWith(template, mapping, DefaultPattern)
 | |
| }
 | |
| 
 | |
| // ExtractVariables returns a map of all the variables defined in the specified
 | |
| // composefile (dict representation) and their default value if any.
 | |
| func ExtractVariables(configDict map[string]interface{}, pattern *regexp.Regexp) map[string]Variable {
 | |
| 	if pattern == nil {
 | |
| 		pattern = DefaultPattern
 | |
| 	}
 | |
| 	return recurseExtract(configDict, pattern)
 | |
| }
 | |
| 
 | |
| func recurseExtract(value interface{}, pattern *regexp.Regexp) map[string]Variable {
 | |
| 	m := map[string]Variable{}
 | |
| 
 | |
| 	switch value := value.(type) {
 | |
| 	case string:
 | |
| 		if values, is := extractVariable(value, pattern); is {
 | |
| 			for _, v := range values {
 | |
| 				m[v.Name] = v
 | |
| 			}
 | |
| 		}
 | |
| 	case map[string]interface{}:
 | |
| 		for _, elem := range value {
 | |
| 			submap := recurseExtract(elem, pattern)
 | |
| 			for key, value := range submap {
 | |
| 				m[key] = value
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 	case []interface{}:
 | |
| 		for _, elem := range value {
 | |
| 			if values, is := extractVariable(elem, pattern); is {
 | |
| 				for _, v := range values {
 | |
| 					m[v.Name] = v
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return m
 | |
| }
 | |
| 
 | |
| type Variable struct {
 | |
| 	Name          string
 | |
| 	DefaultValue  string
 | |
| 	PresenceValue string
 | |
| 	Required      bool
 | |
| }
 | |
| 
 | |
| func extractVariable(value interface{}, pattern *regexp.Regexp) ([]Variable, bool) {
 | |
| 	sValue, ok := value.(string)
 | |
| 	if !ok {
 | |
| 		return []Variable{}, false
 | |
| 	}
 | |
| 	matches := pattern.FindAllStringSubmatch(sValue, -1)
 | |
| 	if len(matches) == 0 {
 | |
| 		return []Variable{}, false
 | |
| 	}
 | |
| 	values := []Variable{}
 | |
| 	for _, match := range matches {
 | |
| 		groups := matchGroups(match, pattern)
 | |
| 		if escaped := groups[groupEscaped]; escaped != "" {
 | |
| 			continue
 | |
| 		}
 | |
| 		val := groups[groupNamed]
 | |
| 		if val == "" {
 | |
| 			val = groups[groupBraced]
 | |
| 		}
 | |
| 		name := val
 | |
| 		var defaultValue string
 | |
| 		var presenceValue string
 | |
| 		var required bool
 | |
| 		switch {
 | |
| 		case strings.Contains(val, ":?"):
 | |
| 			name, _ = partition(val, ":?")
 | |
| 			required = true
 | |
| 		case strings.Contains(val, "?"):
 | |
| 			name, _ = partition(val, "?")
 | |
| 			required = true
 | |
| 		case strings.Contains(val, ":-"):
 | |
| 			name, defaultValue = partition(val, ":-")
 | |
| 		case strings.Contains(val, "-"):
 | |
| 			name, defaultValue = partition(val, "-")
 | |
| 		case strings.Contains(val, ":+"):
 | |
| 			name, presenceValue = partition(val, ":+")
 | |
| 		case strings.Contains(val, "+"):
 | |
| 			name, presenceValue = partition(val, "+")
 | |
| 		}
 | |
| 		values = append(values, Variable{
 | |
| 			Name:          name,
 | |
| 			DefaultValue:  defaultValue,
 | |
| 			PresenceValue: presenceValue,
 | |
| 			Required:      required,
 | |
| 		})
 | |
| 	}
 | |
| 	return values, len(values) > 0
 | |
| }
 | |
| 
 | |
| // Soft default (fall back if unset or empty)
 | |
| func defaultWhenEmptyOrUnset(substitution string, mapping Mapping) (string, bool, error) {
 | |
| 	return withDefaultWhenAbsence(substitution, mapping, true)
 | |
| }
 | |
| 
 | |
| // Hard default (fall back if-and-only-if empty)
 | |
| func defaultWhenUnset(substitution string, mapping Mapping) (string, bool, error) {
 | |
| 	return withDefaultWhenAbsence(substitution, mapping, false)
 | |
| }
 | |
| 
 | |
| func defaultWhenNotEmpty(substitution string, mapping Mapping) (string, bool, error) {
 | |
| 	return withDefaultWhenPresence(substitution, mapping, true)
 | |
| }
 | |
| 
 | |
| func defaultWhenSet(substitution string, mapping Mapping) (string, bool, error) {
 | |
| 	return withDefaultWhenPresence(substitution, mapping, false)
 | |
| }
 | |
| 
 | |
| func requiredErrorWhenEmptyOrUnset(substitution string, mapping Mapping) (string, bool, error) {
 | |
| 	return withRequired(substitution, mapping, ":?", func(v string) bool { return v != "" })
 | |
| }
 | |
| 
 | |
| func requiredErrorWhenUnset(substitution string, mapping Mapping) (string, bool, error) {
 | |
| 	return withRequired(substitution, mapping, "?", func(_ string) bool { return true })
 | |
| }
 | |
| 
 | |
| func withDefaultWhenPresence(substitution string, mapping Mapping, notEmpty bool) (string, bool, error) {
 | |
| 	sep := "+"
 | |
| 	if notEmpty {
 | |
| 		sep = ":+"
 | |
| 	}
 | |
| 	if !strings.Contains(substitution, sep) {
 | |
| 		return "", false, nil
 | |
| 	}
 | |
| 	name, defaultValue := partition(substitution, sep)
 | |
| 	defaultValue, err := Substitute(defaultValue, mapping)
 | |
| 	if err != nil {
 | |
| 		return "", false, err
 | |
| 	}
 | |
| 	value, ok := mapping(name)
 | |
| 	if ok && (!notEmpty || (notEmpty && value != "")) {
 | |
| 		return defaultValue, true, nil
 | |
| 	}
 | |
| 	return value, true, nil
 | |
| }
 | |
| 
 | |
| func withDefaultWhenAbsence(substitution string, mapping Mapping, emptyOrUnset bool) (string, bool, error) {
 | |
| 	sep := "-"
 | |
| 	if emptyOrUnset {
 | |
| 		sep = ":-"
 | |
| 	}
 | |
| 	if !strings.Contains(substitution, sep) {
 | |
| 		return "", false, nil
 | |
| 	}
 | |
| 	name, defaultValue := partition(substitution, sep)
 | |
| 	defaultValue, err := Substitute(defaultValue, mapping)
 | |
| 	if err != nil {
 | |
| 		return "", false, err
 | |
| 	}
 | |
| 	value, ok := mapping(name)
 | |
| 	if !ok || (emptyOrUnset && value == "") {
 | |
| 		return defaultValue, true, nil
 | |
| 	}
 | |
| 	return value, true, nil
 | |
| }
 | |
| 
 | |
| func withRequired(substitution string, mapping Mapping, sep string, valid func(string) bool) (string, bool, error) {
 | |
| 	if !strings.Contains(substitution, sep) {
 | |
| 		return "", false, nil
 | |
| 	}
 | |
| 	name, errorMessage := partition(substitution, sep)
 | |
| 	errorMessage, err := Substitute(errorMessage, mapping)
 | |
| 	if err != nil {
 | |
| 		return "", false, err
 | |
| 	}
 | |
| 	value, ok := mapping(name)
 | |
| 	if !ok || !valid(value) {
 | |
| 		return "", true, &MissingRequiredError{
 | |
| 			Reason:   errorMessage,
 | |
| 			Variable: name,
 | |
| 		}
 | |
| 	}
 | |
| 	return value, true, nil
 | |
| }
 | |
| 
 | |
| func matchGroups(matches []string, pattern *regexp.Regexp) map[string]string {
 | |
| 	groups := make(map[string]string)
 | |
| 	for i, name := range pattern.SubexpNames()[1:] {
 | |
| 		groups[name] = matches[i+1]
 | |
| 	}
 | |
| 	return groups
 | |
| }
 | |
| 
 | |
| // Split the string at the first occurrence of sep, and return the part before the separator,
 | |
| // and the part after the separator.
 | |
| //
 | |
| // If the separator is not found, return the string itself, followed by an empty string.
 | |
| func partition(s, sep string) (string, string) {
 | |
| 	if strings.Contains(s, sep) {
 | |
| 		parts := strings.SplitN(s, sep, 2)
 | |
| 		return parts[0], parts[1]
 | |
| 	}
 | |
| 	return s, ""
 | |
| }
 |