使用JQ扁平化嵌套的Json对象

时间:2018-08-20 06:13:43

标签: json jq flatten keypaths

我正在创建一个脚本,将JSON对象转换为“字符串”文件(用于翻译)。这个想法是要转变的:

{
   "TRANSLATION1": "text1",
   "TRANSLATION2": "text2"
}

进入

"TRANSLATION1" = "text1";
"TRANSLATION2" = "text2";

这是通过: jq -r 'to_entries|map("\"\(.key)\"=\(.value|tojson);")|.[]'

很好!

现在,我的问题是嵌套对象:

{
   "TRANSLATION1": "text1",
   "TRANSLATION2": "text2",
   "TRANSLATION3": {
     "SUBTRANS1": "subtranslation1",
     "SUBTRANS2": "subtranslation2",
   }
}

我希望得到这样的结果:

"TRANSLATION1" = "text1";
"TRANSLATION2" = "text2";
"TRANSLATION3.SUBTRANS1" = "subtranslation1";
"TRANSLATION3.SUBTRANS2" = "subtranslation2";

任何人都可以帮忙吗?我已经挠头好几个小时了...

1 个答案:

答案 0 :(得分:3)

一种方法是使用tostream

tostream
| select(length==2)
| (.[0] | join(".")) as $k
| .[1] as $v
| "\"\($k)\" = \"\($v)\";"

与-r命令行选项一起使用时,如果输入是有效的JSON,这将产生所需的结果。

检查关键假设[*]

可能有必要明确指出,如果任何键名包含一个句点,则输出格式可能不是很有用,因此可能值得检查的确是这种情况,例如如下:

[.. | objects | keys_unsorted[]]
| map(select(index(".")))
| unique[]

如果您的jq没有tostream

paths as $path
| getpath($path) 
| strings
| "\"\($path|join("."))\" = \"\(.)\";"

[*]双关语意