一尘不染

使用jq将x = y对变成键/值对

json

我正在尝试从的JSON输出解析环境变量docker inspect。令人讨厌的是,这些环境变量没有作为有用的键值对返回。它们只是x =
y字符串的数组。这是输出的相关代码段:

[
    {
        "Config": {
            "Env": [
                "JENKINS_HOST=1.2.3.4",
                "JENKINS_INSTANCE=tea",
                "JENKINS_NAME=Enterprise Architecture Tools",
                "JENKINS_VERSION=2.46.2",
                "JENKINS_PROTOCOL=http"
            ]
        }
    }
]

我想将该数组转换为这样的东西:

{
  "Config": {
    "Env": {
      "JENKINS_HOST": "1.2.3.4",
      "JENKINS_INSTANCE": "tea",
      "JENKINS_NAME": "Enterprise Architecture Tools",
      "JENKINS_VERSION": "2.46.2",
      "JENKINS_PROTOCOL": "http"
    }
  }
}

这样,我可以使用一条命令jq '.[] | .Config.Env.JENKINS_HOST'来获取我关心的值。我不知道如何做到这一点。

选择数据甚至将键和值拆分为单独的元素相对容易。例如,如果使用jq '.[] | .Config.Env | .[] | split("=")',我将获得如下数据:

[
  "JENKINS_HOST",
  "1.2.3.4"
]
[
  "JENKINS_INSTANCE",
  "tea"
]
[
  "JENKINS_NAME",
  "Enterprise Architecture Tools"
]
[
  "JENKINS_VERSION",
  "2.46.2"
]
[
  "JENKINS_PROTOCOL",
  "http"
]

但是,我不知道如何将数据转换为对象分配。似乎应该是map或的某种组合reduce,但是我很困惑。谁能指出我正确的方向?


阅读 216

收藏
2020-07-27

共1个答案

一尘不染

要将两个字符串(例如[“ k”,“ v”])的数组转换为对象,可以编写:

{ (.[0]) : .[1] }

因此,您需要编写类似以下内容的内容:

 map(.Config.Env |= (map( split("=") | { (.[0]) : .[1] } ) | add))

2点

抽象出数组到对象的功能使解决方案更易于理解:

def a2o: map( split("=") | { (.[0]) : .[1] } ) | add;

map(.Config.Env |= a2o)

使用matchcapture代替split

由于可能在每个var=value字符串的“值”部分出现“
=”字符,因此split天真地使用可能不是一个好主意。假设您的jq支持正则表达式,这是一个更可靠的选择:

match("([^=]*)=(.*)") | .captures | {(.[0].string) : .[1].string}

或者,更简洁一些,也可能更优雅一些:

[capture( "(?<key>[^:]*):(?<value>.*)" )] | from_entries

index/1

如果您的jq不支持正则表达式,则可以index/1按照以下方式使用:

index("=") as $ix | {(.[:$ix]) : .[$ix+1:]}
2020-07-27