如何覆盖 属性 中 ElasticSearch.NET 的类型和数据
How can I override a property's type and datat in ElasticSearch.NET
我有一个这样的对象:
public class MyDate {
public DateTime Original { get; set; }
public string MungedFormatForAnotherSystem { get; set; }
}
public class ESClass {
public string Id { get;set; }
public MyDate LastUpdatedDate {get;set;}
}
我是这样使用 NEST 客户端的:
var client = new ElasticClient();
client.Index(doc, i => i.Index("myindex");
我想要的是将 MyDate 作为 Original
部分序列化到 ES 中,然后对其进行查询以执行相同的操作。
由于依赖关系,我无法更改对象(如果我必须更改依赖关系,我会这样做,但会尽量避免这种情况)。
有什么办法可以实现,还是一厢情愿?
您可以使用自定义 Json.Net JsonConverter
void Main()
{
var pool = new SingleNodeConnectionPool(new Uri("http://localhost:9200"));
var defaultIndex = "default-index";
var connectionSettings = new ConnectionSettings(pool)
.DefaultIndex(defaultIndex);
var client = new ElasticClient(connectionSettings);
client.CreateIndex(defaultIndex, c => c
.Mappings(m => m
.Map<ESClass>(mm => mm
.AutoMap()
.Properties(p => p
// map the type as a "date" field type
.Date(t => t
.Name(n => n.LastUpdatedDate)
)
)
)
)
);
client.IndexMany(new[] {
new ESClass { Id = "1", LastUpdatedDate = new MyDate { Original = DateTime.UtcNow.Date } },
new ESClass { Id = "2", LastUpdatedDate = new MyDate() },
new ESClass { Id = "3" }
});
client.Refresh(defaultIndex);
// returns document with Id "1"
client.Search<ESClass>(s => s
.Query(q => +q
.Term(f => f.LastUpdatedDate, DateTime.UtcNow.Date)
)
);
}
public class ESClass
{
public string Id { get; set; }
public MyDate LastUpdatedDate { get; set; }
}
[JsonConverter(typeof(MyDateConverter))]
public class MyDate
{
public DateTime Original { get; set; }
public string MungedFormatForAnotherSystem { get; set; }
}
public class MyDateConverter : JsonConverter
{
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
var v = (MyDate)value;
writer.WriteValue(v.Original);
}
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
switch (reader.TokenType)
{
case JsonToken.Date:
return new MyDate { Original = (DateTime)reader.Value };
default:
throw new JsonSerializationException($"Cannot deserialize {reader.TokenType} to {typeof(MyDate).FullName}");
}
}
public override bool CanConvert(Type objectType) => typeof(MyDate).IsAssignableFrom(objectType);
}
我有一个这样的对象:
public class MyDate {
public DateTime Original { get; set; }
public string MungedFormatForAnotherSystem { get; set; }
}
public class ESClass {
public string Id { get;set; }
public MyDate LastUpdatedDate {get;set;}
}
我是这样使用 NEST 客户端的:
var client = new ElasticClient();
client.Index(doc, i => i.Index("myindex");
我想要的是将 MyDate 作为 Original
部分序列化到 ES 中,然后对其进行查询以执行相同的操作。
由于依赖关系,我无法更改对象(如果我必须更改依赖关系,我会这样做,但会尽量避免这种情况)。
有什么办法可以实现,还是一厢情愿?
您可以使用自定义 Json.Net JsonConverter
void Main()
{
var pool = new SingleNodeConnectionPool(new Uri("http://localhost:9200"));
var defaultIndex = "default-index";
var connectionSettings = new ConnectionSettings(pool)
.DefaultIndex(defaultIndex);
var client = new ElasticClient(connectionSettings);
client.CreateIndex(defaultIndex, c => c
.Mappings(m => m
.Map<ESClass>(mm => mm
.AutoMap()
.Properties(p => p
// map the type as a "date" field type
.Date(t => t
.Name(n => n.LastUpdatedDate)
)
)
)
)
);
client.IndexMany(new[] {
new ESClass { Id = "1", LastUpdatedDate = new MyDate { Original = DateTime.UtcNow.Date } },
new ESClass { Id = "2", LastUpdatedDate = new MyDate() },
new ESClass { Id = "3" }
});
client.Refresh(defaultIndex);
// returns document with Id "1"
client.Search<ESClass>(s => s
.Query(q => +q
.Term(f => f.LastUpdatedDate, DateTime.UtcNow.Date)
)
);
}
public class ESClass
{
public string Id { get; set; }
public MyDate LastUpdatedDate { get; set; }
}
[JsonConverter(typeof(MyDateConverter))]
public class MyDate
{
public DateTime Original { get; set; }
public string MungedFormatForAnotherSystem { get; set; }
}
public class MyDateConverter : JsonConverter
{
public override void WriteJson(JsonWriter writer, object value, JsonSerializer serializer)
{
var v = (MyDate)value;
writer.WriteValue(v.Original);
}
public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
switch (reader.TokenType)
{
case JsonToken.Date:
return new MyDate { Original = (DateTime)reader.Value };
default:
throw new JsonSerializationException($"Cannot deserialize {reader.TokenType} to {typeof(MyDate).FullName}");
}
}
public override bool CanConvert(Type objectType) => typeof(MyDate).IsAssignableFrom(objectType);
}