mirror of
https://github.com/mikefarah/yq.git
synced 2025-01-13 03:45:37 +00:00
fa6fac1a76
* Remove extra backtick * Reword explanation of update * Reword explanation of relative update * Change "remaple" to "remain" * Change "clovver" to "clobber" * Reword explanation of update for comment operators * Reword explanation of relative update for comment operators * Change "array" to "expression" * Change "the golangs" to "Golang's" * Change "golangs" to "Golang's" * Change "can durations" to "can add durations" * Change "array scalars" to "arrays" * Change "beit" to "be it" * Fix typo in `eval` tip * Fix typo in header for `has` operation * Add space before pipe in `line` operator example * Fix typos in explanation of deep array merges * Change "is now used" to "is now used." * Change "object," to "object." * Changes "indexes" to "indices" * Remove extraneous copied text from `..` article * Reword explanation of `...` operator * Change "your are" to "you are" * Add link to `string` operator docs in `select` article * Change "is a" to "parameter specifies" in `string` operators article * Change "new line" to "newline" * Change "golang regex" to "Golang's regex" * Change "golang" to "Golang" * Add period * Remove comma in `subtract` article * Remove duplicate number subtraction example * Remove comma in `traverse` operator article * Clarify use of brackets when `read`ing with special characters
249 lines
6.0 KiB
Go
249 lines
6.0 KiB
Go
package yqlib
|
|
|
|
import (
|
|
"testing"
|
|
)
|
|
|
|
var mergeAnchorAssign = `a: &a
|
|
x: OriginalValue
|
|
b:
|
|
<<: *a`
|
|
|
|
var assignOperatorScenarios = []expressionScenario{
|
|
{
|
|
description: "Create yaml file",
|
|
expression: `.a.b = "cat" | .x = "frog"`,
|
|
expected: []string{
|
|
"D0, P[], ()::a:\n b: cat\nx: frog\n",
|
|
},
|
|
},
|
|
{
|
|
skipDoc: true,
|
|
document: "{}",
|
|
expression: `.a |= .b`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::a: null\n",
|
|
},
|
|
},
|
|
{
|
|
skipDoc: true,
|
|
document: mergeAnchorAssign,
|
|
expression: `.c = .b | .a.x = "ModifiedValue" | explode(.)`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::a:\n x: ModifiedValue\nb:\n x: ModifiedValue\nc:\n x: ModifiedValue\n",
|
|
},
|
|
},
|
|
{
|
|
skipDoc: true,
|
|
document: "{}",
|
|
expression: `.a = .b`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::a: null\n",
|
|
},
|
|
},
|
|
{
|
|
skipDoc: true,
|
|
description: "self reference",
|
|
document: "a: cat",
|
|
expression: `.a = [.a]`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::a:\n - cat\n",
|
|
},
|
|
},
|
|
{
|
|
skipDoc: true,
|
|
description: "change to number when old value is valid number",
|
|
document: `a: "3"`,
|
|
expression: `.a = 3`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::a: 3\n",
|
|
},
|
|
},
|
|
{
|
|
skipDoc: true,
|
|
description: "change to bool when old value is valid bool",
|
|
document: `a: "true"`,
|
|
expression: `.a = true`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::a: true\n",
|
|
},
|
|
},
|
|
{
|
|
skipDoc: true,
|
|
description: "update custom tag string, dont clobber style",
|
|
document: `a: !cat "meow"`,
|
|
expression: `.a = "woof"`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::a: !cat \"woof\"\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Update node to be the child value",
|
|
document: `{a: {b: {g: foof}}}`,
|
|
expression: `.a |= .b`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::{a: {g: foof}}\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Double elements in an array",
|
|
document: `[1,2,3]`,
|
|
expression: `.[] |= . * 2`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::[2, 4, 6]\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Update node from another file",
|
|
subdescription: "Note this will also work when the second file is a scalar (string/number)",
|
|
document: `{a: apples}`,
|
|
document2: "{b: bob}",
|
|
expression: `select(fileIndex==0).a = select(fileIndex==1) | select(fileIndex==0)`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::{a: {b: bob}}\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Update node to be the sibling value",
|
|
document: `{a: {b: child}, b: sibling}`,
|
|
expression: `.a = .b`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::{a: sibling, b: sibling}\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Updated multiple paths",
|
|
document: `{a: fieldA, b: fieldB, c: fieldC}`,
|
|
expression: `(.a, .c) = "potato"`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::{a: potato, b: fieldB, c: potato}\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Update string value",
|
|
document: `{a: {b: apple}}`,
|
|
expression: `.a.b = "frog"`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::{a: {b: frog}}\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Update string value via |=",
|
|
subdescription: "Note there is no difference between `=` and `|=` when the RHS is a scalar",
|
|
document: `{a: {b: apple}}`,
|
|
expression: `.a.b |= "frog"`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::{a: {b: frog}}\n",
|
|
},
|
|
},
|
|
{
|
|
skipDoc: true,
|
|
document: `{a: {b: apple}}`,
|
|
expression: `.a.b | (. |= "frog")`,
|
|
expected: []string{
|
|
"D0, P[a b], (!!str)::frog\n",
|
|
},
|
|
},
|
|
{
|
|
skipDoc: true,
|
|
document: `{a: {b: apple}}`,
|
|
expression: `.a.b |= 5`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::{a: {b: 5}}\n",
|
|
},
|
|
},
|
|
{
|
|
skipDoc: true,
|
|
document: `{a: {b: apple}}`,
|
|
expression: `.a.b |= 3.142`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::{a: {b: 3.142}}\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Update deeply selected results",
|
|
subdescription: "Note that the LHS is wrapped in brackets! This is to ensure we don't first filter out the yaml and then update the snippet.",
|
|
document: `{a: {b: apple, c: cactus}}`,
|
|
expression: `(.a[] | select(. == "apple")) = "frog"`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::{a: {b: frog, c: cactus}}\n",
|
|
},
|
|
},
|
|
{
|
|
skipDoc: true,
|
|
document: `{a: {b: apple, c: cactus}}`,
|
|
expression: `(.a.[] | select(. == "apple")) = "frog"`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::{a: {b: frog, c: cactus}}\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Update array values",
|
|
document: `[candy, apple, sandy]`,
|
|
expression: `(.[] | select(. == "*andy")) = "bogs"`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::[bogs, apple, bogs]\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Update empty object",
|
|
dontFormatInputForDoc: true,
|
|
document: `{}`,
|
|
expression: `.a.b |= "bogs"`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::a:\n b: bogs\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Update node value that has an anchor",
|
|
subdescription: "Anchor will remain",
|
|
dontFormatInputForDoc: true,
|
|
document: `a: &cool cat`,
|
|
expression: `.a = "dog"`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::a: &cool dog\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Update empty object and array",
|
|
dontFormatInputForDoc: true,
|
|
document: `{}`,
|
|
expression: `.a.b.[0] |= "bogs"`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::a:\n b:\n - bogs\n",
|
|
},
|
|
},
|
|
{
|
|
skipDoc: true,
|
|
document: `{}`,
|
|
expression: `.a.b.[1].c |= "bogs"`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::a:\n b:\n - null\n - c: bogs\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Custom types are maintained by default",
|
|
document: "a: !cat meow\nb: !dog woof",
|
|
expression: `.a = .b`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::a: !cat woof\nb: !dog woof\n",
|
|
},
|
|
},
|
|
{
|
|
description: "Custom types: clobber",
|
|
subdescription: "Use the `c` option to clobber custom tags",
|
|
document: "a: !cat meow\nb: !dog woof",
|
|
expression: `.a =c .b`,
|
|
expected: []string{
|
|
"D0, P[], (doc)::a: !dog woof\nb: !dog woof\n",
|
|
},
|
|
},
|
|
}
|
|
|
|
func TestAssignOperatorScenarios(t *testing.T) {
|
|
for _, tt := range assignOperatorScenarios {
|
|
testScenario(t, &tt)
|
|
}
|
|
documentOperatorScenarios(t, "assign-update", assignOperatorScenarios)
|
|
}
|