2018-02-17 16:12:41 -02:00
|
|
|
package templater
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2024-05-16 16:20:59 +01:00
|
|
|
"fmt"
|
2023-08-29 21:04:01 +00:00
|
|
|
"maps"
|
2020-05-16 15:45:41 -03:00
|
|
|
"strings"
|
2018-02-17 16:12:41 -02:00
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
"github.com/go-task/task/v3/internal/deepcopy"
|
2023-12-29 20:32:03 +00:00
|
|
|
"github.com/go-task/task/v3/taskfile/ast"
|
2024-04-24 20:47:24 +01:00
|
|
|
"github.com/go-task/template"
|
2018-02-17 16:12:41 -02:00
|
|
|
)
|
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
// Cache is a help struct that allow us to call "replaceX" funcs multiple
|
2018-02-17 16:12:41 -02:00
|
|
|
// times, without having to check for error each time. The first error that
|
|
|
|
// happen will be assigned to r.err, and consecutive calls to funcs will just
|
|
|
|
// return the zero value.
|
2024-03-10 17:11:07 +00:00
|
|
|
type Cache struct {
|
2023-12-29 20:32:03 +00:00
|
|
|
Vars *ast.Vars
|
2018-02-17 16:12:41 -02:00
|
|
|
|
2023-03-30 20:03:59 +00:00
|
|
|
cacheMap map[string]any
|
2019-08-25 13:16:59 -07:00
|
|
|
err error
|
2018-02-17 16:12:41 -02:00
|
|
|
}
|
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
func (r *Cache) ResetCache() {
|
2019-08-25 13:16:59 -07:00
|
|
|
r.cacheMap = r.Vars.ToCacheMap()
|
2019-08-25 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
func (r *Cache) Err() error {
|
|
|
|
return r.err
|
2023-06-15 15:04:03 +00:00
|
|
|
}
|
|
|
|
|
2024-04-24 20:47:24 +01:00
|
|
|
func ResolveRef(ref string, cache *Cache) any {
|
|
|
|
// If there is already an error, do nothing
|
|
|
|
if cache.err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize the cache map if it's not already initialized
|
|
|
|
if cache.cacheMap == nil {
|
|
|
|
cache.cacheMap = cache.Vars.ToCacheMap()
|
|
|
|
}
|
|
|
|
|
2024-05-16 16:20:59 +01:00
|
|
|
if ref == "." {
|
|
|
|
return cache.cacheMap
|
|
|
|
}
|
|
|
|
t, err := template.New("resolver").Funcs(templateFuncs).Parse(fmt.Sprintf("{{%s}}", ref))
|
|
|
|
if err != nil {
|
|
|
|
cache.err = err
|
|
|
|
return nil
|
|
|
|
}
|
2024-06-09 20:30:43 +00:00
|
|
|
val, err := t.Resolve(cache.cacheMap)
|
2024-04-24 20:47:24 +01:00
|
|
|
if err != nil {
|
|
|
|
cache.err = err
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return val
|
|
|
|
}
|
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
func Replace[T any](v T, cache *Cache) T {
|
|
|
|
return ReplaceWithExtra(v, cache, nil)
|
2023-06-15 15:04:03 +00:00
|
|
|
}
|
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
func ReplaceWithExtra[T any](v T, cache *Cache, extra map[string]any) T {
|
|
|
|
// If there is already an error, do nothing
|
|
|
|
if cache.err != nil {
|
|
|
|
return v
|
2018-02-17 16:12:41 -02:00
|
|
|
}
|
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
// Initialize the cache map if it's not already initialized
|
|
|
|
if cache.cacheMap == nil {
|
|
|
|
cache.cacheMap = cache.Vars.ToCacheMap()
|
2018-02-17 16:12:41 -02:00
|
|
|
}
|
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
// Create a copy of the cache map to avoid editing the original
|
|
|
|
// If there is extra data, merge it with the cache map
|
|
|
|
data := maps.Clone(cache.cacheMap)
|
|
|
|
if extra != nil {
|
|
|
|
maps.Copy(data, extra)
|
2018-02-17 16:12:41 -02:00
|
|
|
}
|
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
// Traverse the value and parse any template variables
|
|
|
|
copy, err := deepcopy.TraverseStringsFunc(v, func(v string) (string, error) {
|
|
|
|
tpl, err := template.New("").Funcs(templateFuncs).Parse(v)
|
|
|
|
if err != nil {
|
|
|
|
return v, err
|
|
|
|
}
|
|
|
|
var b bytes.Buffer
|
|
|
|
if err := tpl.Execute(&b, data); err != nil {
|
|
|
|
return v, err
|
|
|
|
}
|
|
|
|
return strings.ReplaceAll(b.String(), "<no value>", ""), nil
|
|
|
|
})
|
2023-06-15 15:04:03 +00:00
|
|
|
if err != nil {
|
2024-03-10 17:11:07 +00:00
|
|
|
cache.err = err
|
|
|
|
return v
|
2018-02-17 16:12:41 -02:00
|
|
|
}
|
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
return copy
|
2018-02-17 16:12:41 -02:00
|
|
|
}
|
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
func ReplaceGlobs(globs []*ast.Glob, cache *Cache) []*ast.Glob {
|
|
|
|
if cache.err != nil || len(globs) == 0 {
|
2023-11-29 19:38:12 -06:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-12-29 20:32:03 +00:00
|
|
|
new := make([]*ast.Glob, len(globs))
|
2023-11-29 19:38:12 -06:00
|
|
|
for i, g := range globs {
|
2023-12-29 20:32:03 +00:00
|
|
|
new[i] = &ast.Glob{
|
2024-03-10 17:11:07 +00:00
|
|
|
Glob: Replace(g.Glob, cache),
|
2023-11-29 19:38:12 -06:00
|
|
|
Negate: g.Negate,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return new
|
|
|
|
}
|
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
func ReplaceVar(v ast.Var, cache *Cache) ast.Var {
|
|
|
|
return ReplaceVarWithExtra(v, cache, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func ReplaceVarWithExtra(v ast.Var, cache *Cache, extra map[string]any) ast.Var {
|
2024-04-24 20:47:24 +01:00
|
|
|
if v.Ref != "" {
|
|
|
|
return ast.Var{Value: ResolveRef(v.Ref, cache)}
|
|
|
|
}
|
2024-03-10 17:11:07 +00:00
|
|
|
return ast.Var{
|
|
|
|
Value: ReplaceWithExtra(v.Value, cache, extra),
|
|
|
|
Sh: ReplaceWithExtra(v.Sh, cache, extra),
|
|
|
|
Live: v.Live,
|
|
|
|
Ref: v.Ref,
|
|
|
|
Dir: v.Dir,
|
|
|
|
}
|
2023-06-15 15:04:03 +00:00
|
|
|
}
|
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
func ReplaceVars(vars *ast.Vars, cache *Cache) *ast.Vars {
|
|
|
|
return ReplaceVarsWithExtra(vars, cache, nil)
|
2023-06-15 15:04:03 +00:00
|
|
|
}
|
|
|
|
|
2024-03-10 17:11:07 +00:00
|
|
|
func ReplaceVarsWithExtra(vars *ast.Vars, cache *Cache, extra map[string]any) *ast.Vars {
|
|
|
|
if cache.err != nil || vars.Len() == 0 {
|
2018-02-17 16:12:41 -02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-12-29 20:32:03 +00:00
|
|
|
var newVars ast.Vars
|
|
|
|
_ = vars.Range(func(k string, v ast.Var) error {
|
2024-03-10 17:11:07 +00:00
|
|
|
newVars.Set(k, ReplaceVarWithExtra(v, cache, extra))
|
2020-03-29 16:54:59 -03:00
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
2023-11-29 16:21:21 +00:00
|
|
|
return &newVars
|
2018-02-17 16:12:41 -02:00
|
|
|
}
|