yq/pkg/yqlib/operator_variables.go

73 lines
2.1 KiB
Go
Raw Normal View History

2021-02-03 04:51:26 +00:00
package yqlib
import (
"container/list"
"fmt"
)
func getVariableOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
variableName := expressionNode.Operation.StringValue
log.Debug("getVariableOperator %v", variableName)
result := context.GetVariable(variableName)
if result == nil {
result = list.New()
}
return context.ChildContext(result), nil
}
2021-09-12 06:55:55 +00:00
type assignVarPreferences struct {
IsReference bool
}
2023-02-23 05:14:36 +00:00
func useWithPipe(d *dataTreeNavigator, context Context, originalExp *ExpressionNode) (Context, error) {
return Context{}, fmt.Errorf("must use variable with a pipe, e.g. `exp as $x | ...`")
}
func variableLoop(d *dataTreeNavigator, context Context, originalExp *ExpressionNode) (Context, error) {
log.Debug("variable loop!")
variableExp := originalExp.LHS
lhs, err := d.GetMatchingNodes(context.ReadOnlyClone(), variableExp.LHS)
2021-02-03 04:51:26 +00:00
if err != nil {
2022-02-09 00:47:21 +00:00
return Context{}, err
2021-02-03 04:51:26 +00:00
}
2023-02-23 05:14:36 +00:00
if variableExp.RHS.Operation.OperationType.Type != "GET_VARIABLE" {
2021-02-03 04:51:26 +00:00
return Context{}, fmt.Errorf("RHS of 'as' operator must be a variable name e.g. $foo")
}
2023-02-23 05:14:36 +00:00
variableName := variableExp.RHS.Operation.StringValue
prefs := variableExp.Operation.Preferences.(assignVarPreferences)
results := list.New()
2021-09-12 06:55:55 +00:00
2023-02-23 05:14:36 +00:00
// now we loop over lhs, set variable to each result and calculate originalExp.Rhs
for el := lhs.MatchingNodes.Front(); el != nil; el = el.Next() {
var variableValue = list.New()
if prefs.IsReference {
variableValue.PushBack(el.Value)
} else {
copy, err := el.Value.(*CandidateNode).Copy()
if err != nil {
return Context{}, err
}
variableValue.PushBack(copy)
}
log.Debug("PROCESSING VARIABLE: ", NodeToString(el.Value.(*CandidateNode)))
newContext := context.ChildContext(context.MatchingNodes)
newContext.SetVariable(variableName, variableValue)
2021-09-12 06:55:55 +00:00
2023-02-23 05:14:36 +00:00
rhs, err := d.GetMatchingNodes(newContext, originalExp.RHS)
if err != nil {
return Context{}, err
}
results.PushBackList(rhs.MatchingNodes)
2021-09-12 06:55:55 +00:00
}
2023-02-23 05:14:36 +00:00
// if there is no LHS - then I guess we just calculate originalExp.Rhs
if lhs.MatchingNodes.Len() == 0 {
return d.GetMatchingNodes(context, originalExp.RHS)
}
return context.ChildContext(results), nil
2021-02-03 04:51:26 +00:00
}