更改 JSON 列表中匹配条件的条目而不丢弃文档的其余部分
Change entry in a JSON list that matches a condition without discarding rest of document
我正在尝试打开一个文件,查看该文件并根据该值更改一个值,然后将其传递给一个文件或 var。
以下是 JSON
的示例
{
"Par": [
{
"Key": "12345L",
"Value": "https://100.100.100.100:100",
"UseLastValue": true
},
{
"Key": "12345S",
"Value": "VAL2CHANGE",
"UseLastValue": true
},
{
"Key": "12345T",
"Value": "HAPPY-HELLO",
"UseLastValue": true
}
],
"CANCOPY": false,
"LOGFILE": ["HELPLOG"]
}
我一直在使用jq并且我已经成功地隔离了对象组并更改了值。
cat jsonfile,json | jq '.Par | map(select(.Value=="VAL2CHANGE")) | .[] | .Value="VALHASBEENCHANGED"'
这给出了
{
"Key": "12345S",
"Value": "VALHASBEENCHANGED",
"UseLastValue": true
}
id 想要实现的是保留具有更改值的完整 JSON 输出
{
"Par": [
{
"Key": "12345L",
"Value": "https://100.100.100.100:100",
"UseLastValue": true
},
{
"Key": "12345S",
"Value": "VALHASBEENCHANGED",
"UseLastValue": true
},
{
"Key": "12345T",
"Value": "HAPPY-HELLO",
"UseLastValue": true
}
],
"CANCOPY": false,
"LOGFILE": ["HELPLOG"]
}
即
jq '.Par | map(select(.Value=="VAL2CHANGE")) | .[] | .Value="VALHASBEENCHANGED"' (NOW PUT IT BACK IN FILE)
或
打开文件,查看文件,要更改的文件值,更改并输出到文件或屏幕
要补充一点,json 文件将只包含我在创建它时查找一次的值。如果任何其他值需要更改,我将以不同的方式命名。
jq --arg match "VAL2CHANGE" \
--arg replace "VALHASBEENCHANGED" \
'.Par |= map(if .Value == $match then (.Value=$replace) else . end)' \
<in.json
为了更全面地替换嵌套数据结构中任何位置的字符串,您可以使用 walk
函数——该函数将包含在 jq 1.6 的标准库中,但可以手动拉入在 1.5 中:
jq --arg match "VAL2CHANGE" \
--arg replace "VALHASBEENCHANGED" '
# taken from jq 1.6; will not be needed here after that version is released.
# Apply f to composite entities recursively, and to atoms
def walk(f):
. as $in
| if type == "object" then
reduce keys_unsorted[] as $key
( {}; . + { ($key): ($in[$key] | walk(f)) } ) | f
elif type == "array" then map( walk(f) ) | f
else f
end;
walk(if . == $match then $replace else . end)' <in.json
如果您只是根据值进行替换,您可以流式传输文件并在重建结果时替换值。
$ jq --arg change 'VAL2CHANGE' --arg value 'VALHASBEENCHANGED' -n --stream '
fromstream(inputs | if length == 2 and .[1] == $change then .[1] = $value else . end)
' input.json
我正在尝试打开一个文件,查看该文件并根据该值更改一个值,然后将其传递给一个文件或 var。 以下是 JSON
的示例{
"Par": [
{
"Key": "12345L",
"Value": "https://100.100.100.100:100",
"UseLastValue": true
},
{
"Key": "12345S",
"Value": "VAL2CHANGE",
"UseLastValue": true
},
{
"Key": "12345T",
"Value": "HAPPY-HELLO",
"UseLastValue": true
}
],
"CANCOPY": false,
"LOGFILE": ["HELPLOG"]
}
我一直在使用jq并且我已经成功地隔离了对象组并更改了值。
cat jsonfile,json | jq '.Par | map(select(.Value=="VAL2CHANGE")) | .[] | .Value="VALHASBEENCHANGED"'
这给出了
{
"Key": "12345S",
"Value": "VALHASBEENCHANGED",
"UseLastValue": true
}
id 想要实现的是保留具有更改值的完整 JSON 输出
{
"Par": [
{
"Key": "12345L",
"Value": "https://100.100.100.100:100",
"UseLastValue": true
},
{
"Key": "12345S",
"Value": "VALHASBEENCHANGED",
"UseLastValue": true
},
{
"Key": "12345T",
"Value": "HAPPY-HELLO",
"UseLastValue": true
}
],
"CANCOPY": false,
"LOGFILE": ["HELPLOG"]
}
即
jq '.Par | map(select(.Value=="VAL2CHANGE")) | .[] | .Value="VALHASBEENCHANGED"' (NOW PUT IT BACK IN FILE)
或
打开文件,查看文件,要更改的文件值,更改并输出到文件或屏幕
要补充一点,json 文件将只包含我在创建它时查找一次的值。如果任何其他值需要更改,我将以不同的方式命名。
jq --arg match "VAL2CHANGE" \
--arg replace "VALHASBEENCHANGED" \
'.Par |= map(if .Value == $match then (.Value=$replace) else . end)' \
<in.json
为了更全面地替换嵌套数据结构中任何位置的字符串,您可以使用 walk
函数——该函数将包含在 jq 1.6 的标准库中,但可以手动拉入在 1.5 中:
jq --arg match "VAL2CHANGE" \
--arg replace "VALHASBEENCHANGED" '
# taken from jq 1.6; will not be needed here after that version is released.
# Apply f to composite entities recursively, and to atoms
def walk(f):
. as $in
| if type == "object" then
reduce keys_unsorted[] as $key
( {}; . + { ($key): ($in[$key] | walk(f)) } ) | f
elif type == "array" then map( walk(f) ) | f
else f
end;
walk(if . == $match then $replace else . end)' <in.json
如果您只是根据值进行替换,您可以流式传输文件并在重建结果时替换值。
$ jq --arg change 'VAL2CHANGE' --arg value 'VALHASBEENCHANGED' -n --stream '
fromstream(inputs | if length == 2 and .[1] == $change then .[1] = $value else . end)
' input.json