在 shell 脚本中生成 JSON

在 shell 脚本中生成 JSON

我需要生成 JSON 输出,如下所示:

[{
    "Service": "service1-name",
    "AWS Account": "service1-dev",
    "AD Accounts": {
        "APP-Service1-Admin": ["a", "b"],
        "APP-Service1-View": ["c", "d"]
    }
}
]

我正在尝试这个 shell 脚本,但无法插入内部信息

#!/bin/bash
innerAD=$(jq -n --arg aaaa aaaa \
              --arg xxxx "xxxx" \
              '$ARGS.named'
)
inner=$(jq -n --argjson "APP-Service1-Admin" "[$innerAD]" \
              '$ARGS.named'
)
final=$(jq -n --arg Service "service1-name" \
              --arg "AWS Account" "service1-dev" \
              --argjson "AD Accounts" "[$inner]" \
              '$ARGS.named'
)
echo "$final"

答案1

一种建议是使用--argswithjq创建两个数组,然后将它们收集到主文档中的正确位置。请注意,这--args必须是命令行上的最后一个选项,并且全部其余的命令行参数将成为$ARGS.positional数组的元素。

{
    jq -n --arg key APP-Service1-Admin '{($key): $ARGS.positional}' --args a b
    jq -n --arg key APP-Service1-View  '{($key): $ARGS.positional}' --args c d
} |
jq -s --arg key 'AD Accounts' '{($key): add}' |
jq --arg Service service1-name --arg 'AWS account' service1-dev '$ARGS.named + .'

前两次jq调用创建一组两个 JSON 对象:

{
  "APP-Service1-Admin": [
    "a",
    "b"
  ]
}
{
  "APP-Service1-View": [
    "c",
    "d"
  ]
}

第三次jq调用用于-s将该集合读入数组,该数组在传递时成为合并对象add。合并的对象被分配给我们的顶级键:

{
  "AD Accounts": {
    "APP-Service1-Admin": [
      "a",
      "b"
    ],
    "APP-Service1-View": [
      "c",
      "d"
    ]
  }
}

最后jq将剩余的顶级键及其值添加到对象中:

{
  "Service": "service1-name",
  "AWS account": "service1-dev",
  "AD Accounts": {
    "APP-Service1-Admin": [
      "a",
      "b"
    ],
    "APP-Service1-View": [
      "c",
      "d"
    ]
  }
}

jo

jo -d . \
    Service=service1-name \
    'AWS account'=service1-dev  \
    'AD Accounts.APP-Service1-Admin'="$(jo -a a b)" \
    'AD Accounts.APP-Service1-View'="$(jo -a c d)"

“内部”对象是使用.-notation(通过 启用-d .)和几个用于创建数组的命令替换来创建的。

或者您可以删除-d .并使用数组表示法的形式:

jo  Service=service1-name \
    'AWS account'=service1-dev \
    'AD Account[APP-Service1-Admin]'="$(jo -a a b)" \
    'AD Account[APP-Service1-View]'="$(jo -a c d)"

答案2

在 bash 中创建复杂的 json 对象时,我经常使用heredocs:

service=$(thing-what-gets-service)
account=$(thing-what-gets-account)
admin=$(jo -a $(thing-what-gets-admin))
view=$(jo -a $(thing-what-gets-view))

read -rd '' json <<EOF
[
    {
        "Service": "$service",
        "AWS Account": "$account",
        "AD Accounts": {
            "APP-Service1-Admin": $admin,
            "APP-Service1-View": $view
        }
    }
]
EOF

这用于jo创建数组,因为这是一种非常简单的方法,但如果需要,可以以不同的方式完成。

答案3

另一个jo答案:

# 2 shell arrays
admins=("a" "b")
views=("c" "d")

accounts=$(jo "${admins[@]/#/APP_Service1-Admin[]=}" "${views[@]/#/APP_Service1-View[]=}")

final=$(jo "Service=service1-name" "AWS Account=service1-dev" "AD Accounts=$accounts")

echo "$final" | jq .
{
  "Service": "service1-name",
  "AWS Account": "service1-dev",
  "AD Accounts": {
    "APP_Service1-Admin": [
      "a",
      "b"
    ],
    "APP_Service1-View": [
      "c",
      "d"
    ]
  }
}

这是棘手的一点:"${admins[@]/#/APP_Service1-Admin[]=}"

$ printf '%s\n' "${admins[@]/#/APP_Service1-Admin[]=}"
APP_Service1-Admin[]=a
APP_Service1-Admin[]=b

这使您可以从 shell 数组构建一个列表。

答案4

xo(1)是在命令行上生成 JSON 的一个很棒的实用程序。它来自于libxo由瞻博网络开发的项目。方便的是,它已预安装在FreeBSD 自 11.0-RELEASE 起并可以输出 JSON 和 XML 等其他格式。这是许多 FreeBSD 管理工具使用的机制(例如,附注(1))以支持机器友好的输出格式。

以下是手册页的输出示例。

$ xo --style text "The {k:name} weighs {:weight/%d} pounds.\n" fish 6
The fish weighs 6 pounds.

$ xo --style xml "The {k:name} weighs {:weight/%d} pounds.\n" fish 6
<name>fish</name>
<weight>6</weight>

$ xo --style json "The {k:name} weighs {:weight/%d} pounds.\n" fish 6
"name": "fish",
"weight": 6
"name": "fish",

$ xo --style html "The {k:name} weighs {:weight/%d} pounds.\n" fish 6
<div class="line">
     <div class="text">The </div>
     <div class="data" data-tag="name">fish</div>
     <div class="text"> weighs </div>
     <div class="data" data-tag="weight">6</div>
     <div class="text"> pounds.</div>
</div>

相关内容