使用NEST索引动态对象

Ell*_*dil 10 c# elasticsearch asp.net-web-api nest

我正在构建一个API应用程序,它基本上允许用户构建一个文档,该文档可以根据需要进行结构化,并存储在Elasticsearch中.从本质上讲,我为用户提供了一个简单的界面来访问我们的Elasticsearch实例.我正在努力使实现尽可能简单.这是我到目前为止所处理的问题.

预期身体的对象:

public class DocumentModel
{
    public string Index { get; set; }
    public string Type { get; set; }
    public string Id { get; set; }
    [ElasticProperty(Type = FieldType.Nested)]
    public dynamic Document { get; set; }
}
Run Code Online (Sandbox Code Playgroud)

简单实施:

[HttpPost]
[Route("")]
public IHttpActionResult Post(DocumentModel document)
{
    Uri nodeLocation = new Uri("http://localhost:9200");
    IConnectionPool connectionPool = new SniffingConnectionPool(new List<Uri> { nodeLocation });
    ConnectionSettings settings = new ConnectionSettings(connectionPool);
    ElasticClient esClient = new ElasticClient(settings);

    IIndexResponse result = esClient.Index(document, i => i
        .Index(document.Index)
        .Type(document.Type)
        .Id(document.Id));

    return Ok(result.IsValid);
}
Run Code Online (Sandbox Code Playgroud)

这样可以正常工作,但它包含源中的索引,类型和ID.我真正想做的只是在索引时提供这三条信息,但实际上只是索引document.Document,它是动态类型.但是,这似乎与Nest不同意,因为它在IDE和编译时抛出错误:

"匿名函数或方法组不能用作动态绑定操作的组成值"

"不能将lambda表达式用作动态调度操作的参数,而无需先将其转换为委托或表达式树类型".

我怎么能索引document.Document?有没有比使用动态类型更好的方法来处理未知结构的传入JSON文档?

Ell*_*dil 20

有几种方法可以做到这一点.

尝试将文档索引为动态类型将不起作用,但您可以通过IndexRequest对象将其索引为对象.

dynamic dynamicDoc = new { /*fill in document format here*/ };
ElasticClient esClient = new ElasticClient(esSettings);

IndexRequest<object> request = new IndexRequest<object>(dynamicDoc)
{
    Index = "someindex",
    Type = "SomeType",
    Id = "someid"
};

esClient.Index<object>(request);
Run Code Online (Sandbox Code Playgroud)

或者如果批量处理文件

List<dynamic> Documents = new List<dynamic>();
//Populate Documents

BulkDescriptor descriptor = new BulkDescriptor();
foreach(var doc in Documents)
{
    descriptor.Index<object>(i => i
        .Index("someindex")
        .Type("SomeType")
        .Id("someid")
        .Document(doc));
}

esClient.Bulk(descriptor);
Run Code Online (Sandbox Code Playgroud)

NEST(或更准确地说,Elasticsearch.Net)还有一个附加到ElasticClient类的.Raw方法变体,它可以索引原始JSON.使用Raw.Index()让我们做这样的事情:

string documentJson = JsonConvert.SerializeObject(document.Document);

ElasticsearchResponse<string> result = esClient.Raw.Index(document.Index, document.Type, document.Id, documentJson);
Run Code Online (Sandbox Code Playgroud)

响应的类型描述符是您期望响应的类型(字符串意味着您将具有序列化的json响应,您可以反序列化并执行某些操作).这允许我们回避整个对象类型问题,并且NEST将文档完全按预期索引到Elasticsearch中.

  • 作为对dynamic的替代,我使用了Dictionary &lt;string,object&gt;或从类继承。警告:如果您从“字典”继承,则NEST不会自动映射文档中的其他属性(而是将其放入字典中)。这对于变量属性也很有效:公共属性进入POCO属性,变量属性进入Data属性(类型为Dictionary &lt;string,object&gt;)。这种批量方法易于使用。不要忘记获取对Bulk的调用结果以检查.Errors等! (2认同)