在ElasticSearch中从数组中删除元素/对象,然后进行匹配查询

zf2sa74q  于 2024-01-06  发布在  ElasticSearch
关注(0)|答案(4)|浏览(154)

我在尝试从elasticsearch中的数组中删除元素/对象时遇到了问题。
这是索引的Map:

{
    "example1": {
        "mappings": {
            "doc": {
                "properties": {
                    "locations": {
                        "type": "geo_point"
                    },
                    "postDate": {
                        "type": "date"
                    },
                    "status": {
                        "type": "long"
                    },
                    "user": {
                        "type": "text",
                        "fields": {
                            "keyword": {
                                "type": "keyword",
                                "ignore_above": 256
                            }
                        }
                    }
                }
            }
        }
    }
}

字符串
这是一个示例文档。

{
    "_index": "example1",
    "_type": "doc",
    "_id": "8036",
    "_score": 1,
    "_source": {
        "user": "kimchy8036",
        "postDate": "2009-11-15T13:12:00",
        "locations": [
            [
                72.79887719999999,
                21.193036000000003
            ],
            [
                -1.8262150000000001,
                51.178881999999994
            ]
        ]
    }
}


使用下面的查询,我可以添加多个位置。

POST /example1/_update_by_query
{
    "query": {
        "match": {
            "_id": "3"
        }
    },
    "script": {
        "lang": "painless",
        "inline": "ctx._source.locations.add(params.newsupp)",
        "params": {
            "newsupp": [
                -74.00,
                41.12121
            ]
        }
    }
}


但是我不能从位置中删除数组对象。我已经尝试了下面的查询,但它不起作用。

POST example1/doc/3/_update
{
    "script": {
        "lang": "painless",
        "inline": "ctx._source.locations.remove(params.tag)",
        "params": {
            "tag": [
                -74.00,
                41.12121
            ]
        }
    }
}


请让我知道我在这里做错了什么。我使用的是弹性版本5.5.2

jyztefdp

jyztefdp1#

在轻松的脚本中,Array.remove()方法通过索引而不是值来删除。
下面是一个在Elasticsearch脚本中通过值删除数组元素的工作示例:

POST objects/_update_by_query
{
    "query": {
        ...  // use regular ES query to remove only in relevant documents
    },
    "script": {
        "source": """
            if (ctx._source[params.array_attribute] != null) {
                for (int i=ctx._source[params.array_attribute].length-1; i>=0; i--) {
                    if (ctx._source[params.array_attribute][i] == params.value_to_remove) {
                        ctx._source[params.array_attribute].remove(i);
                    }
                }
            }
        """,
        "params": {
            "array_attribute": "<NAME_OF_ARRAY_PROPERTY_TO_REMOVE_VALUE_IN>",
            "value_to_remove": "<VALUE_TO_REMOVE_FROM_ARRAY>",
        }
    }
}

字符串
如果你的脚本只需要从一个特定的数组属性中删除值,你可能需要简化脚本。例如,从文档的.color_list数组中删除"green"

_doc/001 = {
    "color_list": ["red", "blue", "green"]
}


删除"green"的脚本:

POST objects/_update_by_query
{
    "query": {
        ...  // use regular ES query to remove only in relevant documents
    },
    "script": {
        "source": """
            for (int i=ctx._source.color_list.length-1; i>=0; i--) {
                if (ctx._source.color_list[i] == params.color_to_remove) {
                    ctx._source.color_list.remove(i);
                }
            }
        """,
        "params": {
            "color_to_remove": "green"
        }
    }
}

bqjvbblv

bqjvbblv2#

add()不同,remove()获取元素的索引并将其删除。
painless中的ctx._source.locations是一个ArrayList,它有Listremove()方法:
E remove(int index)
在列表中指定的位置添加元素(可选操作).
其他方法见无痛API -列表。
参见this answer示例代码。

wb1gzix0

wb1gzix03#

"script" : {
    "lang":"painless",
    "inline":"ctx._source.locations.remove(params.tag)",
    "params":{
      "tag":indexToRemove
    }
}

字符串
如果使用ctx._source.locations.add(elt)添加元素,则使用ctx._source.locations.remove(indexToRemove)删除数组中元素的索引。

kd3sttzy

kd3sttzy4#

POST /your_index/_update_by_query
{
  "script": {
    "source": "ctx._source.your_array.removeIf(item -> item == params.value)",
    "lang": "painless",
    "params": {
      "value": "value_to_remove"
    }
  },
  "query": {
    "term": {
      "your_array": "value_to_remove"
    }
  }
}

字符串

相关问题