/* 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 patternString = fmt.Sprintf( "%s(?i:(?P%s)|(?P%s)|{(?:(?P%s)}|(?P)))", delimiter, delimiter, substitutionNamed, substitutionBraced, ) 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["escaped"]; escaped != "" { return escaped, true, nil } braced := false substitution := groups["named"] if substitution == "" { substitution = groups["braced"] 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 strings.HasPrefix(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["escaped"]; escaped != "" { continue } val := groups["named"] if val == "" { val = groups["braced"] } 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, "" }