使用通过命令行传递的路径访问JQ中的嵌套值



如何或者是否可能从指定路径作为命令参数的JSON结构中提取数据。我从一些更大的脚本中获得了这个简单的片段,只是为了简单起见,并且有问题要解决:

#!/bin/bash
DATA='{
"level1": {
"level2": {
"level3": {
"foo": "bar",
"bar": "baz",
"baz": "bar"
}
}
}
}'
field="level1.level2.level3"
# does not work
jq -r --arg f ${field} '.[$f] | to_entries | .[] | """ + .key + ""="" + .value + """' <<< ${DATA}
# works
jq -r --arg f ${field} '.level1.level2.level3 | to_entries | .[] | """ + .key + ""="" + .value + """' <<< ${DATA}
# also works                              
field2="level3"
jq -r --arg f ${field2} '.level1.level2 | .[$f] | to_entries | .[] | """ + .key + ""="" + .value + """' <<< ${DATA}

给出如下输出:

user@astra:~/test$ ./jqtest 
jq: error (at <stdin>:12): null (null) has no keys
"foo"="bar"
"bar"="baz"
"baz"="bar"
"foo"="bar"
"bar"="baz"
"baz"="bar"

我做错了什么?

在本例中,.[$f]表示"返回与名为level1.level2.level3的键相关联的值"。看到:

$ jq --arg f 'level1.level2.level3' '.[$f]' <<< '{ "level1.level2.level3": "foo" }'
"foo"

除非任何路径组件包含一个点,否则用点分割$f并使用结果作为getpath的参数应该可以工作。

getpath($f / ".")

jq似乎对输入参数中的路径很挑剔。

一个解决方案是提供路径作为json数组,然后使用getpath将其转换为路径:
field='["level1", "level2", "level3"]'
jq -r --argjson f "$field" 'getpath($f)' <<< ${DATA}

或者针对您的具体问题:


field='["level1", "level2", "level3"]'
jq -r --argjson f "${field}" 'getpath($f) | to_entries | .[] | """ + .key + ""="" + .value + """' <<< ${DATA}

最新更新