反向GREP后,使用JQ解析并返回正确的JSON 替代

问题描述

我一直在努力挣扎,似乎很难挣扎。

我经常整天使用iOs快捷方式监视我的一台服务器,并且在处理JSON对象时,快捷方式应用非常挑剔。

我需要返回一个格式完美(无嵌套)的数组;即:

{
  {
    "term": "alpha"
  },{
    "term": "beta"
  },{
    "term": "gamma"
  },}

数据是从内部API调用获取的,我需要从内部调用中筛选出值。 我想知道从API返回的哪些对象当前不正在运行的同名tmux。

这是API调用的非格式化返回:

{
  "online_terms": [
    {
      "term": "alpha"
    },{
      "term": "beta"
    },{
      "term": "gamma"
    }
  ],"total": 3,}

这是目前使用的单线纸:

list=$(tmux list-sessions -F \#S); curl https://myapi.call| jq -r '.online_terms[] | .term' | grep -v $list

$ list将包含当前正在运行的tmux会话的多行LS样式名称。即:

alpha
omega
kappa

到目前为止,这就是我一直在使用的,但是充其量是古怪的。为了澄清,将我所有的tmux会话声明到 $ list 变量中,然后进行API调用,解析出我需要的嵌套值,对 $ list 进行grep反演。不理想!

它仅返回单行名称,即(过滤出的alpha):

beta
gamma

尽管多次尝试将原始数据回送至JQ,但我仍无法将其解析为上述对象类型。

对于如何更清晰地反转grep tmux ls 值或解析为适当的JSON对象的任何建议,我都表示欢迎,在此先感谢您的帮助!

解决方法

为了展示如何使用jq来解决问题而没有任何黑客手段,我们首先假设黑名单以某种方式可作为文本文件使用,例如blacklist.txt:

alpha
omega
kappa

然后,您可以按照以下几行使用jq的调用:

< blacklist.txt jq --argjson curl "$(curl https://myapi.call)" -Rn '
  [inputs] as $blacklist
  | [ $curl
      | .online_terms[]
      | select( .term | IN($blacklist[]) | not) ]'

在这里,curl的输出已使用--argjson选项提供给jq。

可以轻松修改此解决方案以处理黑名单的其他来源。

替代

您还可以通过STDIN将curl命令的输出提供给jq,但是随后您首先必须将文本黑名单转换为JSON。这可以通过再次调用jq来完成:

curl https://myapi.call | 
  jq --argjson blacklist "$(jq -Rn '[inputs]' blacklist.txt)" '
    [.online_terms[]
     | select( .term | IN($blacklist[]) | not)]'

不好看,但是它突出显示了一个事实,即如果您的黑名单生成器可以将黑名单提供为JSON,事情会更简单。

,

之所以提供以下解决方案,主要是因为它与OP提出的解决方案类似,尽管它使用comm而不是grep,并且假定黑名单位于文本文件blacklist.txt中。 :

curl https://myapi.call |
 jq -r '[.online_terms[].term] | sort[]' |
 comm -23 - <(sort blacklist.txt) |
 jq -Rn '[{term: inputs}]' 

其缺点是需要再次调用jq来重建JSON实体;它也需要两个排序操作,这也许是不希望的,尽管如果已经对黑名单进行了排序,那么当然可以跳过对它的排序。