首页 > 解决方案 > 使用 Nest 运行弹性搜索原始查询

问题描述

我正在尝试使用 Nest.Net 库在弹性搜索上运行原始查询。查询如下:

var json4 = @"
                    {
                      ""query"": {
                        ""bool"": {
                                    ""filter"":{
                                        ""term"":{ ""schoolId"": ""c15677ea-3e1e-4767-936a-2b3c57b00503""}
                                    },
                          ""must"": [
                            {
                              ""multi_match"": {
                                ""query"": ""001 Ali"",
                                ""fields"": [""firstName"",""lastName"", ""phoneNumber"",  ""code"", ""title""],
                                ""type"":  ""cross_fields""
                              }
                    }
                          ]
                        }
                      }
                    }
            ";

        SearchRequest searchRequest;
        using (var stream = new MemoryStream(Encoding.UTF8.GetBytes(json4)))
        {
            searchRequest = client.RequestResponseSerializer.Deserialize<SearchRequest>(stream);
        }

Deserialize 方法引发如下错误:

无法将当前 JSON 对象(例如 {"name":"value"})反序列化为类型“System.Collections.Generic.IEnumerable`1[Nest.QueryContainer]”,因为该类型需要 JSON 数组(例如 [1,2, 3])正确反序列化。要修复此错误,请将 JSON 更改为 JSON 数组(例如 [1,2,3])或更改反序列化类型,使其成为普通的 .NET 类型(例如,不是像整数这样的原始类型,而不是像这样的集合类型可以从 JSON 对象反序列化的数组或列表。JsonObjectAttribute 也可以添加到类型中以强制它从 JSON 对象反序列化。路径“query.bool.filter.term”,第 6 行,位置 51。

查询在 kibana 中运行得很好。

谢谢

标签: elasticsearchnest

解决方案


NEST 反序列化仅支持长形式的查询,即

  • bool查询filter子句必须是查询数组;它不支持传递对象的键是查询的对象
  • term"term": { "field": "value" }不支持查询简写形式;它必须是"term": { "field" : { "value": "value" } }.

以下将起作用

var json4 = @"
    {
      ""query"": {
        ""bool"": {
          ""filter"":[
            { ""term"":{ ""schoolId"": { ""value"": ""c15677ea-3e1e-4767-936a-2b3c57b00503""}} }
          ],
          ""must"": [
            {
              ""multi_match"": {
                ""query"": ""001 Ali"",
                ""fields"": [""firstName"",""lastName"", ""phoneNumber"",  ""code"", ""title""],
                ""type"":  ""cross_fields""
              }
            }
          ]
        }
      }
    }
";

SearchRequest searchRequest;
using (var stream = new MemoryStream(Encoding.UTF8.GetBytes(json4)))
{
    searchRequest = client.Serializer.Deserialize<SearchRequest>(stream);
}

但是.LowLevel,NEST 能够接受 JSON 字符串形式的查询,并使用作为属性公开的低级客户端返回强类型搜索响应这样,就不需要将 JSON 字符串反序列化为 aSearchRequest,只需在提交请求时将其序列化回 JSON。此外,您可以使用您拥有的原始查询

var json4 = @"
    {
      ""query"": {
        ""bool"": {
          ""filter"": { 
            ""term"":{ ""schoolId"": ""c15677ea-3e1e-4767-936a-2b3c57b00503"" } 
          },
          ""must"": [
            {
              ""multi_match"": {
                ""query"": ""001 Ali"",
                ""fields"": [""firstName"",""lastName"", ""phoneNumber"",  ""code"", ""title""],
                ""type"":  ""cross_fields""
              }
            }
          ]
        }
      }
    }
";

client.LowLevel.Search<SearchResponse<object>>("index", "type", json4);

推荐阅读