jq:根据键是否以指定字符串结尾来过滤输入

ssc*_*ssc 3 json key filter jq

我收到了这个JSON数据混乱的消息,我需要提取类型列表:

{
  "token/": {
    "accessor": "auth_token_909d6a81",
    "config": {
      "default_lease_ttl": 0,
      "max_lease_ttl": 0
    },
    "description": "token based credentials",
    "local": false,
    "seal_wrap": false,
    "type": "token"           <-- I need to extract this value ...
  },
  "userpass/": {
    "similar_to": {
      "above": null
    },
    "description": "",
    "local": false,
    "seal_wrap": false,
    "type": "userpass"        <-- ... and this one
  },
  "request_id": "f2a4c135-f699-f29d-ca7c-3320dce0a550",
  "more_keys": "more_values",
  "data": {
    "more_data": {
      "even_more_data": "snipped"
    }
  },
  "you_get_the": "idea"
}
Run Code Online (Sandbox Code Playgroud)

很抱歉,内联注释会在复制和粘贴时弄乱数据,但这似乎是阐明我的目标的最佳方法:对于所有以结尾的根密钥/,我需要的值为.type,因此上述示例的最终结果为token userpass

我设法为根键创建了一个有效的过滤器:

host:~ user$ jq -r '. | keys[] | endswith("/")' <<< "${json_data}"
false
false
false
true
true
false
Run Code Online (Sandbox Code Playgroud)

而且我可以使用该过滤器仅获取所需的键,但这仅是键,而不是它们下面的整个数据结构:

host:~ user$ jq -r '. | keys[] | select(. | endswith("/"))' <<< "${json_data}"
token/
userpass/
Run Code Online (Sandbox Code Playgroud)

我似乎无法将所有这些放在一起...

谁能帮我 ?

pea*_*eak 14

to_entries 值得了解,尤其是因为它通常会导致易于阅读(无变量)的管道,如下所示:

to_entries[] | select(.key|endswith("/")) | .value.type
Run Code Online (Sandbox Code Playgroud)


Rom*_*est 5

jq 解:

jq -r '. as $o | keys_unsorted[] | select(endswith("/")) | $o[.].type' file.json
Run Code Online (Sandbox Code Playgroud)

输出:

token
userpass
Run Code Online (Sandbox Code Playgroud)