无法编写扩展方法以包装NEST客户端方法

时间:2019-02-22 18:54:28

标签: elasticsearch .net-core nest

基本上,我试图编写以下扩展方法,以避免必须不断编写.Suffix("keyword")。我不喜欢到处都是字符串文字,并且我的ElasticSearch索引的多个属性碰巧需要添加关键字后缀才能正确查询。

public static class NestHelperExtensions
{
    public static object UseKeywordSuffix(this object @object)
    {
        return @object.Suffix("keyword");
    }
}

如果我使用上面介绍的方法,则后缀将不适用,但我不确定为什么它不起作用。我可能会以错误的方式进行操作,也许有一种方法可以添加模型属性或某些东西来构建nestCilent.Search<T>,但是当我尝试使用KeywordAttribute时,似乎并没有也可以工作。

任何人都可以提供关于为何并非如此简单的解释,如果可能的话,您可以提供示例或解决方法吗?使用属性?

1 个答案:

答案 0 :(得分:1)

这不起作用,因为Suffix method is specifically handled when visiting the field expression

if (methodCall.Method.Name == nameof(SuffixExtensions.Suffix) && methodCall.Arguments.Any())
{
    VisitConstantOrVariable(methodCall, _stack);
    var callingMember = new ReadOnlyCollection<Expression>(
        new List<Expression> { { methodCall.Arguments.First() } }
    );
    Visit(callingMember);
    return methodCall;
}

因此,像上面这样的扩展方法必须被称为Suffix,并且它必须至少具有一个参数。您可能会想到提供一个默认值为“ keyword”的可选参数会起作用,但是表达式树不支持此功能,因此不起作用。

另一种方法是利用AppendSuffix上的Expression<Func<T, object>>扩展方法来构建一些东西。最好的方法是将lambda表达式从流利的调用中拉出并放入变量

public static class NestHelperExtensions
{
    public static Expression<Func<T, object>> KeywordSuffix<T>(this Expression<Func<T, object>> expression)
    {
        return expression.AppendSuffix("keyword");
    }
}

var client = new ElasticClient();

Expression<Func<Person, object>> firstName = f => f.FirstName;  

var searchResponse = client.Search<Person>(s => s
    .Query(q => q
        .Match(m => m
            .Field(firstName.KeywordSuffix())
            .Query("bar")
        )
    )
);

不太好的方式是将lambda表达式强制转换为Expression<Func<T, object>>内联

var searchResponse = client.Search<Person>(s => s
    .Query(q => q
        .Match(m => m
            .Field(((Expression<Func<Person, object>>)(f => f.FirstName)).KeywordSuffix())
            .Query("bar")
        )
    )
);

另一种也许更简单的方法是为字符串"keyword"引入一个常量,并在Suffix扩展方法中使用该常量;避免在各处使用字符串文字。