JQ, как считать в зависимости от условий? - PullRequest
0 голосов
/ 11 февраля 2019

Используя jq, мне нужно получить счет в массиве в зависимости от двух критериев: он ДОЛЖЕН иметь status === 'skipped' && ref.includes(version)

[
  {
    "id": 15484,
    "sha": "52606c8da57984d1243f436e5d12e275db29a6e0",
    "ref": "v1.4.15",
    "status": "canceled"
  },
  {
    "id": 15483,
    "sha": "52606c8da57984d1243f436e5d12e275db29a6e0",
    "ref": "v1.4.15",
    "status": "canceled"
  },
  {
    "id": 15482,
    "sha": "1b4ccc1dc17e9b8ddb24550c5566d2be6b03465e",
    "ref": "dev",
    "status": "success"
  },
  {
    "id": 15481,
    "sha": "5b6ec939739c5a1513634f3b58bf96522917571d",
    "ref": "dev",
    "status": "failed"
  },
  {
    "id": 15480,
    "sha": "ec18d46f491a4645c68388df91fc41455b421e71",
    "ref": "dev",
    "status": "failed"
  },
  {
    "id": 15479,
    "sha": "dd83a6d6e58cc5114aed8016341ab3c5b3ebb702",
    "ref": "dev",
    "status": "failed"
  },
  {
    "id": 15478,
    "sha": "18ccaf4bc37bf65470b2c6ddaa69e5b4018354a7",
    "ref": "dev",
    "status": "success"
  },
  {
    "id": 15477,
    "sha": "f90900d733bce2be3d9ba9db25f8b51296bc6f3f",
    "ref": "dev",
    "status": "failed"
  },
  {
    "id": 15476,
    "sha": "3cf0431a161e6c9ca90e8248af7b4ec39c54bfb1",
    "ref": "dev",
    "status": "failed"
  },
  {
    "id": 15285,
    "sha": "d24b46edc75d8f7308dbef37d7b27625ef70c845",
    "ref": "dev",
    "status": "success"
  },
  {
    "id": 15265,
    "sha": "52606c8da57984d1243f436e5d12e275db29a6e0",
    "ref": "v1.4.15",
    "status": "success"
  },
  {
    "id": 15264,
    "sha": "9a15f8d4c950047f88c642abda506110b9b0bbd7",
    "ref": "v1.4.15-static",
    "status": "skipped"
  },
  {
    "id": 15263,
    "sha": "9a15f8d4c950047f88c642abda506110b9b0bbd7",
    "ref": "v1.4.15-static",
    "status": "skipped"
  },
  {
    "id": 15262,
    "sha": "76451d2401001c4c51b9800d3cdf62e4cdcc86ba",
    "ref": "v1.4.15-no-js",
    "status": "skipped"
  },
  {
    "id": 15261,
    "sha": "76451d2401001c4c51b9800d3cdf62e4cdcc86ba",
    "ref": "v1.4.15-no-js",
    "status": "skipped"
  },
  {
    "id": 15260,
    "sha": "515cd1b00062e9cbce05420036f5ecc7a898a4bd",
    "ref": "v1.4.15-cli",
    "status": "skipped"
  },
  {
    "id": 15259,
    "sha": "515cd1b00062e9cbce05420036f5ecc7a898a4bd",
    "ref": "v1.4.15-cli",
    "status": "skipped"
  },
  {
    "id": 15258,
    "sha": "b67acd3082da795f022fafc304d267d3afd6b736",
    "ref": "v1.4.15-node",
    "status": "skipped"
  },
  {
    "id": 15257,
    "sha": "b67acd3082da795f022fafc304d267d3afd6b736",
    "ref": "v1.4.15-node",
    "status": "skipped"
  },
  {
    "id": 15256,
    "sha": "4da4a788a85d82527ea568fed4f03da193842a80",
    "ref": "v1.4.15-bs-redux-saga-router-dom-intl",
    "status": "skipped"
  }
]

Мы также хотели бы использовать переменную окружения длязапрос:

  • status=skipped
  • ref=v1.4.15

Эта работа, но без параметров переменной среды:

cat test.json | jq '[.[] | select(.status=="skipped") | select(.ref | startswith("v1.4.15"))] | length'

Какэто возможно?

Ответ:

status=skipped; ref=v1.4.15; cat test.json | jq --arg REF "$ref" --arg STATUS "$status" -r '[.[] | select(.status==$STATUS) | select(.ref | startswith($REF))] | length'

Ответы [ 5 ]

0 голосов
/ 21 февраля 2019

Ради полноты, это будет эквивалентный запрос JSONiq:

let $a := [
   (: copy-paste the entire array here in plain JSON syntax --
      omitted for the sake of brevity :)
]
return count(
   for $obj in $a[]
   where $obj.status eq "skipped"
         and
         matches($obj.ref, "ˆv")
   return $obj
)
0 голосов
/ 11 февраля 2019

также, простое альтернативное решение:

bash $ status=skipped
bash $ ref=v1.4.15
bash $ cat test.json | jtc -w"[status]:<$status>: [-1] [ref]:<^$ref.*>R" | wc -l
       9
bash $ 

проработка на jtc пешеходной дорожке (-w)

  1. [status]:<$status>: - сначала вы найдетевсе записи, соответствующие skipped, ограниченные меткой status

  2. [-1] - из всех найденных записей (на шаге 1) вы переходите на 1 уровень вверх (то есть обращаетесь к родителюнайденные записи)

  3. [ref]:<^$ref.*>R - в найденных родительских записях найти первое совпадение, начинающееся с v1.4.15 (с использованием RE) в пределах метки ref

    • затем передать все найденные строки в утилиту wc для подсчета строк

PS.jtc можно найти на github: https://github.com/ldn-softdev/jtc

0 голосов
/ 11 февраля 2019

Используя карту (выберите (...) или эквивалентную, вы можете использовать length, но, как правило, более эффективно использовать общую функцию подсчета, например:

def sigma(s): reduce s as $s (null; .+$s);

sigma(.[] | select(.status=="skipped" and (.ref | startswith("v1.4.15") )) | 1)

Использование оболочкии переменные окружения

Использование переменных оболочки и окружения рассматривается в руководстве по jq, но вкратце, один из способов передачи строковых значений - это использование параметра командной строки --arg, например, по строкам:

jq --arg status "$status" --arg ref "$ref" -f program.jq test.json
0 голосов
/ 11 февраля 2019

Я знаю, jq популярно здесь, но могу ли я предложить xidel?См. http://videlibri.sourceforge.net/xidel.html.
Так же, как jq, это интерпретатор JSON, но помимо JSONiq вы также можете использовать функции XPath / Xquery для выполнения всяких интересных вещей.

В этом списке перечислены все объекты с 2Критерии:

xidel -s test.json -e '$json()[status="skipped" and starts-with(ref,"v1.4.15")]'

Чтобы подсчитать их, просто заключите запрос с помощью функции count():

xidel -s test.json -e 'count($json()[status="skipped" and starts-with(ref,"v1.4.15")])'

Возвращает 9.

С переменными:

status=skipped
ref=v1.4.15

xidel -s test.json -e 'count($json()[status="'$status'" and starts-with(ref,"'$ref'")])'
0 голосов
/ 11 февраля 2019

Используйте функцию length() в конце фильтра, после помещения списка объектов в массив

jq '[.[] | select(.status == "skipped") | select(.ref | test("1\\.4\\.15"))] | length'

, но для простого возврата объектов не используйте логику, чтобы получить длину

jq '[.[] | select(.status == "skipped") | select(.ref | test("1\\.4\\.15"))]'

test() - более мощный способ сопоставить ваше регулярное выражение со строками JSON.startswith() или endswith() не может соответствовать строкам, если они находятся в середине.

Используя переменные,

ref="1\.4\.15"
jq --arg status "$status" --arg ref "$ref" \
    '[.[] | select(.status == $status) | select(.ref | test($ref))]|length' json
...