1
0
mirror of https://github.com/go-task/task.git synced 2025-01-10 04:18:53 +02:00
task/taskfile/ast/var.go

162 lines
3.4 KiB
Go
Raw Normal View History

package ast
import (
"fmt"
2023-11-30 03:56:30 +02:00
"strings"
"gopkg.in/yaml.v3"
"github.com/go-task/task/v3/internal/experiments"
2024-01-04 02:31:24 +02:00
"github.com/go-task/task/v3/internal/omap"
)
// Vars is a string[string] variables map.
type Vars struct {
2024-01-04 02:31:24 +02:00
omap.OrderedMap[string, Var]
}
2019-08-25 22:16:59 +02:00
// ToCacheMap converts Vars to a map containing only the static
// variables
func (vs *Vars) ToCacheMap() (m map[string]any) {
m = make(map[string]any, vs.Len())
_ = vs.Range(func(k string, v Var) error {
if v.Sh != "" {
// Dynamic variable is not yet resolved; trigger
// <no value> to be used in templates.
return nil
}
2019-08-25 22:16:59 +02:00
if v.Live != nil {
m[k] = v.Live
} else {
m[k] = v.Value
2019-08-25 22:16:59 +02:00
}
return nil
})
return
}
// Wrapper around OrderedMap.Set to ensure we don't get nil pointer errors
func (vs *Vars) Range(f func(k string, v Var) error) error {
if vs == nil {
return nil
}
return vs.OrderedMap.Range(f)
}
// Wrapper around OrderedMap.Merge to ensure we don't get nil pointer errors
func (vs *Vars) Merge(other *Vars) {
if vs == nil || other == nil {
return
}
vs.OrderedMap.Merge(other.OrderedMap)
}
// Wrapper around OrderedMap.Len to ensure we don't get nil pointer errors
func (vs *Vars) Len() int {
if vs == nil {
return 0
}
return vs.OrderedMap.Len()
}
// DeepCopy creates a new instance of Vars and copies
// data by value from the source struct.
func (vs *Vars) DeepCopy() *Vars {
if vs == nil {
return nil
}
return &Vars{
OrderedMap: vs.OrderedMap.DeepCopy(),
}
}
// Var represents either a static or dynamic variable.
type Var struct {
Value any
Live any
Sh string
Json string
Yaml string
Dir string
}
func (v *Var) UnmarshalYAML(node *yaml.Node) error {
if experiments.AnyVariables.Enabled {
2023-12-23 06:59:10 +02:00
// This implementation is not backwards-compatible and replaces the 'sh' key with map variables
if experiments.AnyVariables.Value == "1" {
var value any
if err := node.Decode(&value); err != nil {
return err
}
// If the value is a string and it starts with $, then it's a shell command
if str, ok := value.(string); ok {
if str, ok = strings.CutPrefix(str, "$"); ok {
v.Sh = str
return nil
}
}
v.Value = value
return nil
}
2023-12-23 06:59:10 +02:00
// This implementation IS backwards-compatible and keeps the 'sh' key and allows map variables to be added under the `map` key
if experiments.AnyVariables.Value == "2" {
switch node.Kind {
case yaml.MappingNode:
key := node.Content[0].Value
switch key {
case "sh", "map", "json", "yaml":
2023-12-23 06:59:10 +02:00
var m struct {
Sh string
Map any
Json string
Yaml string
2023-12-23 06:59:10 +02:00
}
if err := node.Decode(&m); err != nil {
return err
}
v.Sh = m.Sh
v.Value = m.Map
v.Json = m.Json
v.Yaml = m.Yaml
2023-12-23 06:59:10 +02:00
return nil
default:
return fmt.Errorf(`yaml: line %d: %q is not a valid variable type. Try "sh", "map", "json", "yaml" or using a scalar value`, node.Line, key)
2023-12-23 06:59:10 +02:00
}
default:
var value any
if err := node.Decode(&value); err != nil {
return err
}
v.Value = value
2023-11-30 03:56:30 +02:00
return nil
}
}
}
switch node.Kind {
case yaml.ScalarNode:
var str string
if err := node.Decode(&str); err != nil {
return err
}
v.Value = str
return nil
case yaml.MappingNode:
var sh struct {
Sh string
}
if err := node.Decode(&sh); err != nil {
return err
}
v.Sh = sh.Sh
return nil
}
return fmt.Errorf("yaml: line %d: cannot unmarshal %s into variable", node.Line, node.ShortTag())
}