json使用jq重新映射

时间:2015-12-06 14:58:30

标签: json bash ubuntu jq

我有以下json:

[
  {
    "ip": [
      "8.8.4.4/32",
      "212.40.11.20/32"
    ],
    "port": 25,
    "proto": "tcp"
  },
  {
    "ip": [
      "212.40.11.30/32"
    ],
    "port": 3389,
    "proto": "tcp"
  }
]

使用jq我想分隔ip数组中的ip地址,如下所示:

[
  {
    "ip": "8.8.4.4/32",
    "port": 25,
    "proto": "tcp"
  },
  {
    "ip": "212.40.11.20/32",
    "port": 25,
    "proto": "tcp"
  },
  {
    "ip": "212.40.11.30/32",
    "port": 3389,
    "proto": "tcp"
  }
]

我在jq 1.5 Ubuntu 15.04上使用bash 4.3.42(1)

编辑:
这就是我从AWS生成此列表的方式:
aws ec2 describe-security-groups --group-id sg-2cf5e31 --query 'SecurityGroups[].IpPermissions[].{port:ToPort,proto:IpProtocol,ip:IpRanges[].CidrIp[]}' | jq '.[]'

下一步是添加自定义字段。

感谢您的协助。

3 个答案:

答案 0 :(得分:7)

比@ peak和@ hek2mgl更简单,更简单的替代方案,但仍然从根本上运用相同的概念:

jq 'map(.ip = .ip[])'

答案 1 :(得分:4)

$ jq 'map(  (.ip[] | { "ip": .}) + del(.ip) )' input.json
[
  {
    "ip": "8.8.4.4/32",
    "port": 25,
    "proto": "tcp"
  },
  {
    "ip": "212.40.11.20/32",
    "port": 25,
    "proto": "tcp"
  },
  {
    "ip": "212.40.11.30/32",
    "port": 3389,
    "proto": "tcp"
  }
]

同样,可以写map( . + {ip:.ip[]} )

那就是(这是使jq如此强大的东西之一),如果STREAMn是长度为n的流,则value + STREAMn产生长度为n的流,假设(value + x)有效对于流中的每个项目。

确实,STREAMn + STREAMm会产生"笛卡尔积产品"长度m * n,假设项目可以成对添加。

答案 2 :(得分:2)

除了Peak's nice solution之外,您还可以使用以下表达式:

svn mv

它很好地展示了jq '[.[]|{ip:.ip[],port,proto}]' aws.json 如何“扁平化” ip 数组。

输出:

jq