2020-11-03 23:48:43 +00:00
|
|
|
package yqlib
|
2020-10-17 11:39:01 +00:00
|
|
|
|
2021-12-02 22:23:16 +00:00
|
|
|
type assignPreferences struct {
|
|
|
|
DontOverWriteAnchor bool
|
2022-01-15 04:48:34 +00:00
|
|
|
OnlyWriteNull bool
|
2022-08-29 05:37:25 +00:00
|
|
|
ClobberCustomTags bool
|
2021-12-02 22:23:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func assignUpdateFunc(prefs assignPreferences) crossFunctionCalculation {
|
|
|
|
return func(d *dataTreeNavigator, context Context, lhs *CandidateNode, rhs *CandidateNode) (*CandidateNode, error) {
|
2023-04-09 01:14:51 +00:00
|
|
|
if !prefs.OnlyWriteNull || lhs.Tag == "!!null" {
|
2023-06-07 17:45:42 +00:00
|
|
|
lhs.UpdateFrom(rhs, prefs)
|
2022-01-15 04:48:34 +00:00
|
|
|
}
|
2021-12-02 22:23:16 +00:00
|
|
|
return lhs, nil
|
|
|
|
}
|
2021-07-07 09:22:51 +00:00
|
|
|
}
|
|
|
|
|
2022-08-29 05:37:25 +00:00
|
|
|
// they way *= (multipleAssign) is handled, we set the multiplePrefs
|
|
|
|
// on the node, not assignPrefs. Long story.
|
|
|
|
func getAssignPreferences(preferences interface{}) assignPreferences {
|
|
|
|
prefs := assignPreferences{}
|
|
|
|
|
|
|
|
switch typedPref := preferences.(type) {
|
|
|
|
case assignPreferences:
|
|
|
|
prefs = typedPref
|
|
|
|
case multiplyPreferences:
|
|
|
|
prefs = typedPref.AssignPrefs
|
|
|
|
}
|
|
|
|
return prefs
|
|
|
|
}
|
|
|
|
|
2021-02-02 07:17:59 +00:00
|
|
|
func assignUpdateOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
2022-02-07 00:55:55 +00:00
|
|
|
lhs, err := d.GetMatchingNodes(context, expressionNode.LHS)
|
2020-10-17 11:39:01 +00:00
|
|
|
if err != nil {
|
2021-02-02 07:17:59 +00:00
|
|
|
return Context{}, err
|
2020-10-17 11:39:01 +00:00
|
|
|
}
|
2021-07-07 09:22:51 +00:00
|
|
|
|
2022-08-29 05:37:25 +00:00
|
|
|
prefs := getAssignPreferences(expressionNode.Operation.Preferences)
|
|
|
|
|
|
|
|
log.Debug("assignUpdateOperator prefs: %v", prefs)
|
2021-12-02 22:23:16 +00:00
|
|
|
|
2021-01-12 23:18:53 +00:00
|
|
|
if !expressionNode.Operation.UpdateAssign {
|
2021-07-07 09:22:51 +00:00
|
|
|
// this works because we already ran against LHS with an editable context.
|
2021-12-02 22:23:16 +00:00
|
|
|
_, err := crossFunction(d, context.ReadOnlyClone(), expressionNode, assignUpdateFunc(prefs), false)
|
2021-07-07 09:22:51 +00:00
|
|
|
return context, err
|
2020-11-19 06:08:13 +00:00
|
|
|
}
|
|
|
|
|
2023-10-05 05:29:12 +00:00
|
|
|
//traverse backwards through the context -
|
|
|
|
// like delete, we need to run against the children first.
|
|
|
|
// (e.g. consider when running with expression '.. |= [.]' - we need
|
|
|
|
// to wrap the children first
|
|
|
|
for el := lhs.MatchingNodes.Back(); el != nil; el = el.Prev() {
|
2020-10-17 11:39:01 +00:00
|
|
|
candidate := el.Value.(*CandidateNode)
|
|
|
|
|
2022-02-07 00:55:55 +00:00
|
|
|
rhs, err := d.GetMatchingNodes(context.SingleChildContext(candidate), expressionNode.RHS)
|
2020-10-17 11:39:01 +00:00
|
|
|
|
|
|
|
if err != nil {
|
2021-02-02 07:17:59 +00:00
|
|
|
return Context{}, err
|
2020-10-17 11:39:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// grab the first value
|
2021-02-02 07:17:59 +00:00
|
|
|
first := rhs.MatchingNodes.Front()
|
2020-10-17 11:39:01 +00:00
|
|
|
|
|
|
|
if first != nil {
|
2021-01-01 23:27:32 +00:00
|
|
|
rhsCandidate := first.Value.(*CandidateNode)
|
2023-06-07 17:45:42 +00:00
|
|
|
candidate.UpdateFrom(rhsCandidate, prefs)
|
2020-10-17 11:39:01 +00:00
|
|
|
}
|
|
|
|
}
|
2021-01-11 22:40:37 +00:00
|
|
|
|
2021-02-02 07:17:59 +00:00
|
|
|
return context, nil
|
2020-10-17 11:39:01 +00:00
|
|
|
}
|
2020-10-19 05:14:29 +00:00
|
|
|
|
|
|
|
// does not update content or values
|
2021-02-02 07:17:59 +00:00
|
|
|
func assignAttributesOperator(d *dataTreeNavigator, context Context, expressionNode *ExpressionNode) (Context, error) {
|
2021-02-08 02:58:46 +00:00
|
|
|
log.Debug("getting lhs matching nodes for update")
|
2022-02-07 00:55:55 +00:00
|
|
|
lhs, err := d.GetMatchingNodes(context, expressionNode.LHS)
|
2020-10-19 05:14:29 +00:00
|
|
|
if err != nil {
|
2021-02-02 07:17:59 +00:00
|
|
|
return Context{}, err
|
2020-10-19 05:14:29 +00:00
|
|
|
}
|
2021-02-02 07:17:59 +00:00
|
|
|
for el := lhs.MatchingNodes.Front(); el != nil; el = el.Next() {
|
2020-10-19 05:14:29 +00:00
|
|
|
candidate := el.Value.(*CandidateNode)
|
|
|
|
|
2022-02-07 00:55:55 +00:00
|
|
|
rhs, err := d.GetMatchingNodes(context.SingleReadonlyChildContext(candidate), expressionNode.RHS)
|
2020-10-19 05:14:29 +00:00
|
|
|
|
|
|
|
if err != nil {
|
2021-02-02 07:17:59 +00:00
|
|
|
return Context{}, err
|
2020-10-19 05:14:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// grab the first value
|
2021-02-02 07:17:59 +00:00
|
|
|
first := rhs.MatchingNodes.Front()
|
2020-10-19 05:14:29 +00:00
|
|
|
|
|
|
|
if first != nil {
|
2021-12-02 22:23:16 +00:00
|
|
|
prefs := assignPreferences{}
|
|
|
|
if expressionNode.Operation.Preferences != nil {
|
|
|
|
prefs = expressionNode.Operation.Preferences.(assignPreferences)
|
|
|
|
}
|
2023-04-09 01:14:51 +00:00
|
|
|
if !prefs.OnlyWriteNull || candidate.Tag == "!!null" {
|
2022-01-15 04:48:34 +00:00
|
|
|
candidate.UpdateAttributesFrom(first.Value.(*CandidateNode), prefs)
|
|
|
|
}
|
2020-10-19 05:14:29 +00:00
|
|
|
}
|
|
|
|
}
|
2021-02-02 07:17:59 +00:00
|
|
|
return context, nil
|
2020-10-19 05:14:29 +00:00
|
|
|
}
|