在elasticsearch查询中将child视为父项的字段

时间:2012-08-04 06:48:33

标签: parent-child elasticsearch

我正在阅读elasticsearch的文档,本[页码] [1]讨论了如何使用_parent将子项映射到父类型。

如果我有一个名为email的孩子被称为account的父母:

每种类型的字段:

account (http://localhost:9200/myapp/account/1)
========
id
name
some_other_info
state

email (http://localhost:9200/myapp/email/1?parent=1)
========
id
email
  • 如果name的{​​{1}}提供account的{​​{1}}字段,我该如何搜索email字段和email字段stateaccount

  • 有没有办法让父母拥有的所有孩子(某种类型或任何类型)?

  • 索引子文档时,是否可以将父对象作为对象属性传递给JSON数据,而不是作为查询字符串的一部分?


在尝试了imotov的建议之后,我提出了这个问题:

这是在active

上执行的
http://localhost:9200/myapp/account/_search

问题是以上内容并未向我提供电子邮件匹配的任何帐户。

我想要的效果基本上是这样的:

  • 有一个搜索框
  • 用户开始输入,搜索框会自动填充。
  • 根据{ "query": { "bool": { "must": [ { "prefix": { "name": "a" } }, { "term": { "statuses": "active" } } ], "should": [ { "has_child": { "type": "emailaddress", "query": { "prefix": { "email": "a" } } } } ] } } } 或任何account类型的名称检查用户的查询。
  • 如果匹配emailaddress,则返回它们。如果accounts匹配,则返回其父帐户。
  • 每次搜索最多限制为x(比方说10个)帐户。

所以,我基本上需要emailaddress两种类型之间的搜索并返回父类型的匹配。


测试数据:

OR

imotov的解决方案对我有用。我找到的另一个解决方案是向curl -XPUT http://localhost:9200/test/account/1 -d '{ "name": "John Smith", "statuses": "active" }' curl -XPUT http://localhost:9200/test/account/2 -d '{ "name": "Peter Smith", "statuses": "active" }' curl -XPUT http://localhost:9200/test/account/3 -d '{ "name": "Andy Smith", "statuses": "active" }' //Set up mapping for parent/child relationship curl -XPUT 'http://localhost:9200/test/email/_mapping' -d '{ "emails" : { "_parent" : {"type" : "account"} } }' curl -XPUT http://localhost:9200/test/email/1?parent=1 -d '{ "email": "john@smith.com" }' curl -XPUT http://localhost:9200/test/email/2?parent=1 -d '{ "email": "admin@mycompany.com" }' curl -XPUT http://localhost:9200/test/email/3?parent=1 -d '{ "email": "abcd@efg.com" }' curl -XPUT http://localhost:9200/test/email/4?parent=2 -d '{ "email": "peter@peter.com" }' curl -XPUT http://localhost:9200/test/email/5?parent=3 -d '{ "email": "andy@yahoo.com" }' curl -XPUT http://localhost:9200/test/email/6?parent=3 -d '{ "email": "support@mycompany.com" }' 查询account,然后对结果运行status = active过滤器,并对子类型bool使用has_childprefix过滤器内的name上。

1 个答案:

答案 0 :(得分:21)

elasticsearch和关系数据库之间的一个重要区别是elasticsearch无法执行连接。在elasticsearch中,您始终在搜索单个索引或索引联合。但是在父/子关系的情况下,可以使用子索引上的查询来限制父索引中的结果。例如,您可以在account类型上执行此查询。

{
    "bool": {
        "must": [
            { 
                "text" : { "name": "foo" } 
            }, { 
                "term" : { "state": "active" } 
            }, {
                "has_child": {
                    "type": "email",
                    "query": {
                        "text": {"email": "bar" }
                    }
                }
            }
        ]
    }
}

此查询仅返回父文档(不会返回子文档)。您可以使用此查询返回的父ID来使用字段_parent查找此父项的所有子项,该字段默认存储并编制索引。

{
    "term" : { "_parent": "1" } 
}

或者,您只能将结果限制为字段bar中包含email字词的孩子:

{
    "bool": {
        "must": [
            { 
                "term" : { "_parent": "1" } 
            }, { 
                "text" : { "email": "bar" } 
            }
        ]
    }
}

除非您使用_bulk indexing,否则我认为不可能在json中指定parent。

这是使用问题中提供的测试数据实现电子邮件查询的方式:

#!/bin/sh
curl -XDELETE 'http://localhost:9200/test' && echo 
curl -XPOST 'http://localhost:9200/test' -d '{
    "settings" : {
        "number_of_shards" : 1,
        "number_of_replicas" : 0
    },
    "mappings" : {
      "account" : {
        "_source" : { "enabled" : true },
        "properties" : {
          "name": { "type": "string", "analyzer": "standard" },
          "statuses": { "type": "string",  "index": "not_analyzed" }
        }
      },
      "email" : {
        "_parent" : {
          "type" : "account"
        },
        "properties" : {
          "email": { "type": "string",  "analyzer": "standard" }
        }
      }
    }
}' && echo

curl -XPUT 'http://localhost:9200/test/account/1' -d '{
    "name": "John Smith",
    "statuses": "active"
}'

curl -XPUT 'http://localhost:9200/test/account/2' -d '{
    "name": "Peter Smith",
    "statuses": "active"
}'

curl -XPUT 'http://localhost:9200/test/account/3' -d '{
    "name": "Andy Smith",
    "statuses": "active"
}'

//Set up mapping for parent/child relationship

curl -XPUT 'http://localhost:9200/test/email/1?parent=1' -d '{
    "email": "john@smith.com"
}'

curl -XPUT 'http://localhost:9200/test/email/2?parent=1' -d '{
    "email": "admin@mycompany.com"
}'

curl -XPUT 'http://localhost:9200/test/email/3?parent=1' -d '{
    "email": "abcd@efg.com"
}'

curl -XPUT 'http://localhost:9200/test/email/4?parent=2' -d '{
    "email": "peter@peter.com"
}'

curl -XPUT 'http://localhost:9200/test/email/5?parent=3' -d '{
    "email": "andy@yahoo.com"
}'

curl -XPUT 'http://localhost:9200/test/email/6?parent=3' -d '{
    "email": "support@mycompany.com"
}'

curl -XPOST 'http://localhost:9200/test/_refresh'
echo
curl 'http://localhost:9200/test/account/_search' -d '{
  "query": {
    "bool": {
      "must": [
        {
          "term": {
            "statuses": "active"
          }
        }
      ],
      "should": [
        {
          "prefix": {
            "name": "a"
          }
        },
        {
          "has_child": {
            "type": "email",
            "query": {
              "prefix": {
                "email": "a"
              }
            }
          }
        }
      ],
      "minimum_number_should_match" : 1
    }
  }
}' && echo