使用字符串过滤 JMESPath

Filtering JMESPath with a string

尽管进行了大量研究,我还是找不到解决方案。我坚持使用 contains 功能。 我有这个 Json 文件:

    {
  "from": "Api",
  "success": true,
  "message": "",
  "errors": [],
  "data": {
    "operations": [
      {
        "IDOperation": 100,
        "DateEcriture": "2019-01-02",
        "Comment": "Invoice Nh5 numero 152",
        "sous_operations": []
      },
      {
        "IDOperation": 101,
        "DateEcriture": "2019-01-02",
        "Comment": "one other thing",
        "sous_operations": []
      },
      {
        "IDOperation":102,
        "DateEcriture": "2019-01-02",
        "Comment": "an other thing",
        "sous_operations": [{"ID-sous-Operation": 103,
                           "DateEcriture": "2019-01-02",
                           "Comment": "Invoice Nh15 numero 341"}]
      }]
   } 
}

而且我想过滤 "Comment" 字段中包含单词 "Invoice" 的对象以获得此:

{"operations": [
      {
        "IDOperation": 100,
        "DateEcriture": "2019-01-02",
        "Comment": "Invoice Nh5 numero 152"
      },
      {
        "IDOperation": 103,
        "DateEcriture": "2019-01-02",
        "Comment": "Invoice Nh15 numero 341"
      }]
}

感谢您的帮助

您没有说明您在哪一部分遇到了困难。我猜它正在处理嵌套的子操作,因为这对我来说似乎是最难和最不明显的部分,但我会尽力涵盖所有内容。

这是我的假设:

  • 输入总是包含一个带有字段 data.
  • 的对象
  • data 字段始终是具有字段 operations.
  • 的对象
  • operations 总是一个数组。
  • operations的每个成员都有相同的四个字段:IDOperationDateEcritureCommentsous_operations
  • sous_operations 总是一个数组。
  • sous_operations 的每个成员都有相同的三个字段:ID-sous-Operation (!)、DateEcritureComment
  • 特别是,子操作的嵌套深度不超过一层。
  • 所有名为 Comment 的字段都是字符串。
  • 您想查找在 Comment 字段中具有 "Invoice"(不区分大小写)的操作和子操作。
  • 您想输出它们,但不想输出它们可能具有的任何子操作。
  • 您想将 ID-sous-Operation 重命名为 IDOperation
  • 输出应由一个包含单个字段的对象组成 operations,该字段是选定和转换操作的数组。

我认为这符合您的要求:

{
  operations:
    data.operations|
    map(
      &[
        [
          {
            IDOperation:IDOperation,
            DateEcriture:DateEcriture,
            Comment:Comment            
          }
        ],
        map(
          &{
            IDOperation:"ID-sous-Operation",
            DateEcriture:DateEcriture,
            Comment:Comment
          },
          sous_operations
        )
      ],
      @
    )|
    [][]|
    [?contains(Comment,`"Invoice"`)]
}

首先,我们将每个操作替换为一个二元数组。第一个成员是一个单元素数组,包含操作的字段但不包含其子操作。第二个成员是所有操作的子操作的数组。 (此时我们还重命名了子操作ID字段。)

所以现在我们将操作作为简化操作数组的(双元素)数组的数组。我们两次使用 flatten 运算符来获得一个简单的单级数组。最后我们简单地使用 contains 方法过滤它。

这是输出:

$ jp --filename input1.json --expr-file filter.jmespath
{
  "operations": [
    {
      "Comment": "Invoice Nh5 numero 152",
      "DateEcriture": "2019-01-02",
      "IDOperation": 100
    },
    {
      "Comment": "Invoice Nh15 numero 341",
      "DateEcriture": "2019-01-02",
      "IDOperation": 103
    }
  ]
}