ElasticSearch NEST 5.6.1查询单元测试

时间:2018-06-12 10:08:57

标签: elasticsearch nest

我为弹性搜索写了一堆查询,我想为它们编写一个单元测试。使用这篇文章moq an elastic connection我能够完成一般的嘲弄。但是当我试图查看从我的查询生成的Json时,我没有设法以任何方式获取它。 我尝试关注此帖elsatic query moq,但它仅与旧版本的Nest相关,因为方法ConnectionStatusRequestInformation不再适用于ISearchResponse对象。< / p>

我的测试看起来如下:

[TestMethod]
 public void VerifyElasticFuncJson()
{
//Arrange
var elasticService = new Mock<IElasticService>();
var elasticClient = new Mock<IElasticClient>();
var clinet = new ElasticClient();
var searchResponse = new Mock<ISearchResponse<ElasticLog>>();
elasticService.Setup(es => es.GetConnection())
    .Returns(elasticClient.Object);

elasticClient.Setup(ec => ec.Search(It.IsAny<Func<SearchDescriptor<ElasticLog>, 
                          ISearchRequest>>())).
                          Returns(searchResponse.Object);

//Act
var service = new ElasticCusipInfoQuery(elasticService.Object);
var FindFunc = service.MatchCusip("CusipA", HostName.GSMSIMPAPPR01, 
                                        LogType.Serilog);
var con = GetConnection();
var search =  con.Search<ElasticLog>(sd => sd
             .Type(LogType.Serilog)
             .Index("logstash-*")
             .Query(q => q
             .Bool(b => b
                    .Must(FindFunc)
                    )
               )     
             );
 **HERE I want to get the JSON** and assert it look as expected**
}

还有其他方法可以实现我的要求吗?

1 个答案:

答案 0 :(得分:1)

执行此操作的最佳方法是使用InMemoryConnection捕获请求字节并将其与预期的JSON进行比较。这是NEST的单元测试。像

这样的东西
private static void Main()
{
    var pool = new SingleNodeConnectionPool(new Uri("http://localhost:9200"));
    var connectionSettings = new ConnectionSettings(pool, new InMemoryConnection())
        .DefaultIndex("default")
        .DisableDirectStreaming();

    var client = new ElasticClient(connectionSettings);

    // Act
    var searchResponse = client.Search<Question>(s => s
       .Query(q => (q
         .Match(m => m
               .Field(f => f.Title)
               .Query("Kibana")
         ) || q
         .Match(m => m
               .Field(f => f.Title)
               .Query("Elasticsearch")
               .Boost(2)
         )) && +q
         .Range(t => t
               .Field(f => f.Score)
               .GreaterThan(0)
         )
       )
    );

    var actual = searchResponse.RequestJson();

    var expected = new 
    {
        query = new {
            @bool = new {
                must = new object[] {
                    new {
                        @bool = new {
                            should = new object[] {
                                new {
                                    match = new {
                                        title = new {
                                            query = "Kibana"
                                        }
                                    }
                                },
                                new {
                                    match = new {
                                        title = new {
                                            query = "Elasticsearch",
                                            boost = 2d
                                        }
                                    }
                                }
                            },
                        }
                    },
                    new {
                        @bool = new {
                            filter = new [] {
                                new {
                                    range = new {
                                        score = new {
                                            gt = 0d
                                        }
                                    }
                                }
                            }
                        }
                    }
                }
            }
        }
    };

    // Assert
    Console.WriteLine(JObject.DeepEquals(JToken.FromObject(expected), JToken.Parse(actual)));
}

public static class Extensions
{
    public static string RequestJson(this IResponse response) =>
        Encoding.UTF8.GetString(response.ApiCall.RequestBodyInBytes);
}

我为预期的JSON使用了匿名类型,因为它比转义的JSON字符串更容易使用。

需要注意的一点是,即使JSON对象中存在重复的对象键(只要最后一个键/值匹配),Json.NET的JObject.DeepEquals(...)也将返回true。如果您只是序列化NEST搜索,那么您可能会遇到的问题,但需要注意的事项。

如果您要检查序列化的许多测试,您将需要创建一个ConnectionSettings实例并与所有实例共享,以便您可以利用其中的内部缓存和测试将比在每个测试中实例化一个新实例更快地运行。