【ElasticSearch+NetCore 第二篇】Nest封装

using Elasticsearch.Net;
using Nest;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;

namespace ESStudy.Website
{
    public static class EsUtil
    {
        /// 
        /// 获取ElasticClient
        /// 
        /// ElasticSearch服务器地址
        /// 默认索引名称
        /// 默认类型
        /// 
        public static ElasticClient Client(string url, string defaultIndex = "", string defaultType = "")
        {
            var uri = new Uri(url);
            var setting = new ConnectionSettings(uri);

            if (!string.IsNullOrWhiteSpace(defaultIndex))
            {
                setting.DefaultIndex(defaultIndex);
            }

            if (!string.IsNullOrWhiteSpace(defaultType))
            {
                setting.DefaultTypeName(defaultType);
            }

            return new ElasticClient(setting);
        }

        /// 
        /// 获取ElasticClient
        /// 
        /// ElasticSearch集群地址
        /// 默认索引名称
        /// 默认类型
        /// 
        public static ElasticClient Client(string[] urls, string defaultIndex = "", string defaultType = "")
        {
            var uris = urls.Select(h => new Uri(h)).ToArray();
            var pool = new SniffingConnectionPool(uris);

            var setting = new ConnectionSettings(pool);

            if (!string.IsNullOrWhiteSpace(defaultIndex))
            {
                setting.DefaultIndex(defaultIndex);
            }

            if (!string.IsNullOrWhiteSpace(defaultType))
            {
                setting.DefaultTypeName(defaultType);
            }

            return new ElasticClient(setting);
        }

        /// 
        /// 如果同名索引不存在则创建索引
        /// 
        /// 
        /// ElasticClient实例
        /// 要创建的索引名称
        /// 默认副本数量,如果是单实例,注意改成0
        /// 默认分片数量
        /// 
        public static bool CreateIndex(this ElasticClient client, string indexName = "", int numberOfReplicas = 1, int numberOfShards = 5) where T : class
        {
            if (client.IndexExists(indexName).Exists) return false;

            var indexState = new IndexState
            {
                Settings = new IndexSettings
                {
                    NumberOfReplicas = numberOfReplicas, //副本数
                    NumberOfShards = numberOfShards //分片数
                }
            };

            if (string.IsNullOrWhiteSpace(indexName))
            {
                indexName = typeof(T).Name.ToLower();
            }

            var result = client.CreateIndex(indexName, c => c.InitializeUsing(indexState).Mappings(ms => ms.Map(m => m.AutoMap())));
            return result.Acknowledged;
        }

        /// 
        /// 返回一个正序排列的委托
        /// 
        /// 
        /// 
        /// 
        public static Func, SortDescriptor> Sort(string field) where T : class
        {
            return sd => sd.Ascending(field);
        }

        public static Func, SortDescriptor> Sort(Expressionobject>> field) where T : class
        {
            return sd => sd.Ascending(field);
        }

        /// 
        /// 返回一个倒序排列的委托
        /// 
        /// 
        /// 
        /// 
        public static Func, SortDescriptor> SortDesc(string field) where T : class
        {
            return sd => sd.Descending(field);
        }

        public static Func, SortDescriptor> SortDesc(Expressionobject>> field) where T : class
        {
            return sd => sd.Descending(field);
        }

        /// 
        /// 返回一个Must条件集合
        /// 
        /// 
        /// 
        public static List, QueryContainer>> Must() where T : class 
        {
            return new List, QueryContainer>>();
        }

        public static List, QueryContainer>> Should() where T : class
        {
            return new List, QueryContainer>>();
        }

        /// 
        /// 添加Match子句
        /// 
        /// 
        /// 
        /// 要查询的列
        /// 要查询的关键字
        /// 
        public static void AddMatch(this List, QueryContainer>> musts, string field,
            string value, double? boost = null) where T : class 
        {
            musts.Add(d => d.Match(mq => mq.Field(field).Query(value).Boost(boost)));
        }

        public static void AddMatch(this List, QueryContainer>> musts, 
            Expressionobject>> field, string value) where T : class
        {
            musts.Add(d => d.Match(mq => mq.Field(field).Query(value)));
        }

        /// 
        /// 添加MultiMatch子句
        /// 
        /// 
        /// 
        /// 要查询的列
        /// 要查询的关键字
        public static void AddMultiMatch(this List, QueryContainer>> musts, 
            string[] fields, string value) where T : class
        {
            musts.Add(d => d.MultiMatch(mq => mq.Fields(fields).Query(value)));
        }

        /// 
        /// 添加MultiMatch子句
        /// 
        /// 
        /// 
        /// 例如:f=>new [] {f.xxx, f.xxx}
        /// 要查询的关键字
        public static void AddMultiMatch(this List, QueryContainer>> musts,
            Expressionobject>> fields, string value) where T : class
        {
            musts.Add(d => d.MultiMatch(mq => mq.Fields(fields).Query(value)));
        }

        /// 
        /// 添加大于子句
        /// 
        /// 
        /// 
        /// 要查询的列
        /// 要比较的值
        public static void AddGreaterThan(this List, QueryContainer>> musts, string field,
            double value) where T : class
        {
            musts.Add(d => d.Range(mq => mq.Field(field).GreaterThan(value)));
        }

        public static void AddGreaterThan(this List, QueryContainer>> musts, 
            Expressionobject>> field, double value) where T : class
        {
            musts.Add(d => d.Range(mq => mq.Field(field).GreaterThan(value)));
        }

        /// 
        /// 添加大于等于子句
        /// 
        /// 
        /// 
        /// 要查询的列
        /// 要比较的值
        public static void AddGreaterThanEqual(this List, QueryContainer>> musts, string field,
            double value) where T : class
        {
            musts.Add(d => d.Range(mq => mq.Field(field).GreaterThanOrEquals(value)));
        }

        public static void AddGreaterThanEqual(this List, QueryContainer>> musts, 
            Expressionobject>> field, double value) where T : class
        {
            musts.Add(d => d.Range(mq => mq.Field(field).GreaterThanOrEquals(value)));
        }

        /// 
        /// 添加小于子句
        /// 
        /// 
        /// 
        /// 要查询的列
        /// 要比较的值
        public static void AddLessThan(this List, QueryContainer>> musts, string field,
            double value) where T : class
        {
            musts.Add(d => d.Range(mq => mq.Field(field).LessThan(value)));
        }

        public static void AddLessThan(this List, QueryContainer>> musts, 
            Expressionobject>> field, double value) where T : class
        {
            musts.Add(d => d.Range(mq => mq.Field(field).LessThan(value)));
        }

        /// 
        /// 添加小于等于子句
        /// 
        /// 
        /// 
        /// 要查询的列
        /// 要比较的值
        public static void AddLessThanEqual(this List, QueryContainer>> musts, string field,
            double value) where T : class
        {
            musts.Add(d => d.Range(mq => mq.Field(field).LessThanOrEquals(value)));
        }

        public static void AddLessThanEqual(this List, QueryContainer>> musts, 
            Expressionobject>> field, double value) where T : class
        {
            musts.Add(d => d.Range(mq => mq.Field(field).LessThanOrEquals(value)));
        }

        /// 
        /// 添加一个Term,一个列一个值
        /// 
        /// 
        /// 
        /// 要查询的列
        /// 要比较的值
        public static void AddTerm(this List, QueryContainer>> musts, string field,
            object value) where T : class
        {
            musts.Add(d => d.Term(field, value));
        }

        public static void AddTerm(this List, QueryContainer>> musts, 
            Expressionobject>> field, object value) where T : class
        {
            musts.Add(d => d.Term(field, value));
        }

        /// 
        /// 添加一个Terms,一个列多个值
        /// 
        /// 
        /// 
        /// 
        /// 
        public static void AddTerms(this List, QueryContainer>> musts, string field,
            object[] values) where T : class
        {
            musts.Add(d => d.Terms(tq => tq.Field(field).Terms(values)));
        }

        public static void AddTerms(this List, QueryContainer>> musts, 
            Expressionobject>> field, object[] values) where T : class
        {
            musts.Add(d => d.Terms(tq => tq.Field(field).Terms(values)));
        }
    }
}

使用方法

/// 
        /// 搜索
        /// 
        /// 
        /// 
        /// 
        public IActionResult Index(CourseEsSearchInput input, int pageIndex = 1)
        {
            pageIndex = pageIndex > 0 ? pageIndex : 1;

            //var musts = new List, QueryContainer>>();
            var musts = EsUtil.Must();
            if (!string.IsNullOrWhiteSpace(input.School))
            {
                //musts.Add(c => c.Term(cc => cc.Field("School").Value(input.School)));
                musts.AddMatch("school", input.School);
            }

            if (!string.IsNullOrWhiteSpace(input.Key))
            {
                //musts.Add(c => c.MultiMatch(cc => cc.Fields(ccc => ccc.Fields(ced => new[] {ced.Title, ced.School})).Query(input.Key)));
                musts.Add(c => c.MultiMatch(cc => cc.Query(input.Key).Fields(new[] { "title", "school" })));
            }

            var must2 = EsUtil.Must();

            if (!string.IsNullOrWhiteSpace(input.Ver1))
            {
                //musts.Add(c => c.Term(cc => cc.Ver1, input.Ver1));
                must2.AddTerm("ver1", input.Ver1);
            }

            if (!string.IsNullOrWhiteSpace(input.Ver2))
            {
                //musts.Add(c => c.Term(cc => cc.Field(ced => ced.Ver2).Value(input.Ver2)));
                must2.AddTerm("ver2", input.Ver2);
            }

            if (!string.IsNullOrWhiteSpace(input.Ver3))
            {
                //musts.Add(c => c.Term(cc => cc.Field(ced => ced.Ver3).Value(input.Ver2)));
                must2.AddTerm("ver3", input.Ver3);
            }

            if (input.PriceStart.HasValue)
            {
                //musts.Add(c => c.Range(cc => cc.Field(ccc => ccc.Price).GreaterThan((double)input.PriceStart.Value)));
                must2.AddGreaterThan("price", (double)input.PriceStart.Value);
            }

            if (input.PriceEnd.HasValue)
            {
                //musts.Add(c => c.Range(cc => cc.Field(ccc => ccc.Price).LessThanOrEquals((double)input.PriceEnd.Value)));
                must2.AddLessThanEqual("price", (double)input.PriceEnd.Value);
            }

            var client = EsUtil.Client("http://127.0.0.1:9200", "course", "doc");
            var result = client.Search(sd =>
                sd.Query(qcd => qcd
                        .Bool(cc => cc
                            .Must(musts)
                            .Filter(must2))
                        )
                        .From(10 * (pageIndex - 1))
                        .Take(10)
                    //.Sort(sdd => sdd.Descending("price"))
                    .Sort(EsUtil.Sort(c => c.Price))
            );

            var total = result.Total;
            var data = result.Documents;
            ViewBag.Total = total;
            return View(data);
        }

        /// 
        /// 创建索引
        /// 
        /// 
        public IActionResult CreateIndex()
        {
            var result = EsUtil.Client("http://127.0.0.1:9200").CreateIndex("course");
            if (result)
            {
                return Content("创建成功");
            }
            else
            {
                return Content("创建失败,可能已存在同名索引");
            }
        }

 

你可能感兴趣的:(【ElasticSearch+NetCore 第二篇】Nest封装)