当前位置: 首页 > 知识库问答 >
问题:

MongoTemplate聚合返回“IllegalArgumentException:无效引用”

傅浩漫
2023-03-14

给定具有以下文档的集合:

{
    "_id": {
        "$oid": "5e8f83289d3a48132e9b6e34"
    },
    "resource-id": "my-default-network-map",
    "uri": "http://alto.example.com/networkmap",
    "mappings": [
       {
        "version-tag": "X",
        "address-aggregations": [{
            "pid": "PID1",
            "ipv4-addresses": ["192.0.2.0/24", "198.51.100.0/25"]
        }, {
            "pid": "PID2",
            "ipv4-addresses": ["198.51.100.128/25"]
        }, {
            "pid": "PID3",
            "ipv4-addresses": ["0.0.0.0/0"],
            "ipv6-addresses": ["::/0"]
        }]
       }, 
       {
        "version-tag": "Y",
        "address-aggregations": [{
            "pid": "PID1",
            "ipv4-addresses": ["192.0.2.0/24", "198.51.100.0/25"]
        }, {
            "pid": "PID2",
            "ipv4-addresses": ["198.51.100.128/25"]
        }]
       }],
    "_class": "com.example.restservice.entity.NetworkMapEntity"
}

假设有更多的这些文档带有其他“resource-id”值,我希望返回我之前键入的同一个文档,但是“mappings”数组只包含version-tag=X的对象,“address-aggregations”数组只包含[pidx,pidy,...]中带有“pid”的对象。我目前能想到的最好的方法是以下聚合管道:

[
  {
    '$match': {
      'resource-id': 'my-default-network-map'
    }
  }, {
    '$unwind': {
      'path': '$mappings'
    }
  }, {
    '$match': {
      'mappings.version-tag': 'X'
    }
  }, {
    '$unwind': {
      'path': '$mappings.address-aggregations'
    }
  }, {
    '$match': {
      'mappings.address-aggregations.pid': {
        '$in': [
          'PID1', 'PID2'
        ]
      }
    }
  }, {
    '$group': {
      '_id': '$_id', 
      'resource-id': {
        '$first': '$resource-id'
      }, 
      'uri': {
        '$first': '$uri'
      }, 
      'version-tag': {
        '$first': '$mappings.version-tag'
      }, 
      'address-aggregations': {
        '$push': '$mappings.address-aggregations'
      }
    }
  }, {
    '$project': {
      '_id': 1, 
      'resource-id': 1, 
      'uri': 1, 
      'mappings.version-tag': '$version-tag', 
      'mappings.address-aggregations': '$address-aggregations'
    }
  }, {
    '$group': {
      '_id': '$_id', 
      'resource-id': {
        '$first': '$resource-id'
      }, 
      'uri': {
        '$first': '$uri'
      }, 
      'mappings': {
        '$push': '$mappings'
      }
    }
  }
]

它在mongodb客户端shell上返回了预期的结果。但是,当我将此类对象转换为带有聚合对象的MongoTemplate查询时:

Aggregation aggregation = newAggregation(
  match(Criteria.where("resource-id").is(resourceId)),
  unwind("mappings"),
  match(Criteria.where("mappings.version-tag").is(versionTag)),
  unwind("mappings.address-aggregations"),
  match(Criteria.where("mappings.address-aggregations.pid").in(pids)),
  group("$_id")
    .first("resource-id").as("resource-id")
    .first("uri").as("uri")
    .first("mappings.version-tag").as("version-tag")
    .push("mappings.address-aggregations").as("address-aggregations"),
  project("_id", "resource-id", "uri")
    .and("version-tag").as("mappings.version-tag")
    .and("address-aggregations").as("mappings.address-aggregations"),
  group("_id")
    .first("resource-id").as("resource-id")
    .first("uri").as("uri")
    .push("mappings").as("mappings")
);


AggregationResults<NetworkMapEntity> aggregationResults = mongoTemplate.aggregate(aggregation, "NetworkMaps", NetworkMapEntity.class);

并对其进行测试,则引发以下异常:java.lang.IllegalArgumentException:无效引用'mappings'!

搜索了一下,我发现有类似问题的人说这是SpringFramework.data依赖项的一个bug,但是是一年以前的版本,所以我认为我构造聚合对象可能是问题所在。我是不是在解决方案的Java代码部分做错了什么?

共有1个答案

袁子瑜
2023-03-14

虽然我仍然不明白为什么当前的解决方案不起作用,并且希望了解为什么不起作用,但我将查询更改如下,并且现在似乎可以按预期工作:

Aggregation aggregation = newAggregation(
  match(Criteria.where("resource-id").is(resourceId)),
  unwind("mappings"),
  match(Criteria.where("mappings.version-tag").is(versionTag)),
  unwind("mappings.address-aggregations"),
  match(Criteria.where("mappings.address-aggregations.pid").in(pids)),
  group("_id", "resource-id", "uri")
    .first("mappings.version-tag").as("version-tag")
    .push("mappings.address-aggregations").as("address-aggregations")
  group("_id._id")
    .first("_id.resource-id").as("resource-id")
    .push(
       new BasicDBObject("version-tag", "$version-tag")
             .append("address-aggregations", "$address-aggregations")).as("mappings")
);
 类似资料:
  • 我有一个集合,其中的文档如下所示:

  • 我是Java8流api的新手,我正在寻求一种解决方案,以运行我的对象列表并聚合某些属性,从而最终能够获得该属性类型和所有聚合结果的新列表。 例如,我的列表中有10个person对象,我想要一个基于第一人称年龄的所有年龄差异的列表 在执行流魔术之后,结果应该是int类型的,并且看起来如下所示

  • 问题内容: 这是logcat: 与之关联的行是: 可能是什么问题呢?创建数据库就好了。您还需要查看其他代码吗? 更新: 我非常确定此列存在。我用这个查询数据库: 在LogCat中: 因此该列确实存在。 问题答案: 我发现了一个奇怪的解决方案。在中。你所要做的:

  • 这个问题不是如何通过多个字段进行聚合,我们可以使用子聚合。 如果你知道SQL,我可以给你一个完美的解释: 我们能在Elasticsearch中实现这一点吗? 谢谢。

  • 我必须创建聚合,计算日期范围中包含的文档数量。我的查询如下所示: 间隔:604800000等于7天。 结果,我重新强调了这些: 您可以要求我的buckets从29/12/2016开始,但作为范围查询,不包括此日期。我希望我的桶应该从01/01/2017开始,正如我在范围查询中指出的那样。此问题仅发生在间隔天数大于1的查询中。在任何其他间隔的情况下,它工作得很好。我试了一天,几个月和几个小时,效果很

  • 但邮差回信说: 知道为什么或如何进一步调试它吗?Spring data-elasticsearch在做我不明白的事情吗? 我应该期待这样的事情: