2021-02-07 01:54:36 -08:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2021-04-11 06:24:03 -07:00
|
|
|
"errors"
|
2021-05-26 12:12:58 -07:00
|
|
|
"reflect"
|
|
|
|
"strings"
|
2021-02-07 01:54:36 -08:00
|
|
|
"text/template"
|
2021-02-12 03:35:59 -08:00
|
|
|
|
|
|
|
"github.com/Masterminds/sprig"
|
2021-02-07 01:54:36 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
// Errors to show when the template handling fails
|
|
|
|
invalidTemplate = "invalid template text"
|
|
|
|
incorrectTemplate = "unable to create text based on template"
|
2021-05-26 12:12:58 -07:00
|
|
|
|
|
|
|
templateEnvRegex = `\.Env\.(?P<ENV>[^ \.}]*)`
|
2021-02-07 01:54:36 -08:00
|
|
|
)
|
|
|
|
|
|
|
|
type textTemplate struct {
|
|
|
|
Template string
|
|
|
|
Context interface{}
|
2021-05-26 12:12:58 -07:00
|
|
|
Env environmentInfo
|
2021-02-07 01:54:36 -08:00
|
|
|
}
|
|
|
|
|
2021-04-11 06:24:03 -07:00
|
|
|
func (t *textTemplate) render() (string, error) {
|
2021-02-12 03:35:59 -08:00
|
|
|
tmpl, err := template.New("title").Funcs(sprig.TxtFuncMap()).Parse(t.Template)
|
2021-02-07 01:54:36 -08:00
|
|
|
if err != nil {
|
2021-04-11 06:24:03 -07:00
|
|
|
return "", errors.New(invalidTemplate)
|
2021-02-07 01:54:36 -08:00
|
|
|
}
|
2021-05-26 12:12:58 -07:00
|
|
|
if strings.Contains(t.Template, ".Env") {
|
|
|
|
t.loadEnvVars()
|
|
|
|
}
|
2021-02-07 01:54:36 -08:00
|
|
|
buffer := new(bytes.Buffer)
|
|
|
|
defer buffer.Reset()
|
|
|
|
err = tmpl.Execute(buffer, t.Context)
|
|
|
|
if err != nil {
|
2021-04-11 06:24:03 -07:00
|
|
|
return "", errors.New(incorrectTemplate)
|
2021-02-07 01:54:36 -08:00
|
|
|
}
|
2021-05-26 12:12:58 -07:00
|
|
|
text := buffer.String()
|
|
|
|
// issue with missingkey=zero ignored for map[string]interface{}
|
|
|
|
// https://github.com/golang/go/issues/24963
|
|
|
|
text = strings.ReplaceAll(text, "<no value>", "")
|
|
|
|
return text, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *textTemplate) loadEnvVars() {
|
|
|
|
context := make(map[string]interface{})
|
|
|
|
switch v := t.Context.(type) {
|
|
|
|
case map[string]interface{}:
|
|
|
|
context = v
|
|
|
|
default:
|
|
|
|
// we currently only support structs
|
|
|
|
if !t.isStruct() {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
context = t.structToMap()
|
|
|
|
}
|
|
|
|
envVars := map[string]string{}
|
|
|
|
matches := findAllNamedRegexMatch(templateEnvRegex, t.Template)
|
|
|
|
for _, match := range matches {
|
|
|
|
envVars[match["ENV"]] = t.Env.getenv(match["ENV"])
|
|
|
|
}
|
|
|
|
context["Env"] = envVars
|
|
|
|
t.Context = context
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *textTemplate) isStruct() bool {
|
|
|
|
v := reflect.TypeOf(t.Context)
|
|
|
|
if v == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if v.Kind() == reflect.Ptr {
|
|
|
|
v = v.Elem()
|
|
|
|
}
|
|
|
|
if v.Kind() == reflect.Invalid {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return v.Kind() == reflect.Struct
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *textTemplate) structToMap() map[string]interface{} {
|
|
|
|
context := make(map[string]interface{})
|
|
|
|
v := reflect.ValueOf(t.Context)
|
|
|
|
strct := v.Type()
|
|
|
|
for i := 0; i < strct.NumField(); i++ {
|
|
|
|
sf := strct.Field(i)
|
|
|
|
name := sf.Name
|
|
|
|
value := v.Field(i).Interface()
|
|
|
|
context[name] = value
|
|
|
|
}
|
|
|
|
return context
|
2021-02-07 01:54:36 -08:00
|
|
|
}
|