-
Notifications
You must be signed in to change notification settings - Fork 116
/
Copy pathAutoCompleteClient.cs
197 lines (173 loc) · 8.02 KB
/
AutoCompleteClient.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
// Copyright (c) MASA Stack All rights reserved.
// Licensed under the MIT License. See LICENSE.txt in the project root for license information.
namespace Masa.Contrib.SearchEngine.AutoComplete;
public class AutoCompleteClient : BaseAutoCompleteClient
{
private readonly IElasticClient _elasticClient;
private readonly IMasaElasticClient _client;
private readonly string _indexName;
private readonly Operator _defaultOperator;
private readonly SearchType _defaultSearchType;
private readonly bool _enableMultipleCondition;
public AutoCompleteClient(
IElasticClient elasticClient,
IMasaElasticClient client,
string indexName,
Operator defaultOperator,
SearchType defaultSearchType,
bool enableMultipleCondition)
{
_elasticClient = elasticClient;
_client = client;
_indexName = indexName;
_defaultOperator = defaultOperator;
_defaultSearchType = defaultSearchType;
_enableMultipleCondition = enableMultipleCondition;
}
public override async Task<GetResponse<TAudoCompleteDocument, TValue>> GetAsync<TAudoCompleteDocument, TValue>(
string keyword,
AutoCompleteOptions? options = null,
CancellationToken cancellationToken = default)
{
var newOptions = options ?? new(_defaultSearchType);
var searchType = newOptions.SearchType ?? _defaultSearchType;
keyword = keyword.Trim();
if (string.IsNullOrEmpty(keyword))
return new GetResponse<TAudoCompleteDocument, TValue>(true, string.Empty, new List<TAudoCompleteDocument>());
if (searchType == SearchType.Fuzzy)
{
var ret = await _client.GetPaginatedListAsync(
new PaginatedOptions<TAudoCompleteDocument>(
_indexName,
GetFuzzyKeyword(keyword),
newOptions.Field,
newOptions.Page,
newOptions.PageSize,
_defaultOperator)
, cancellationToken);
return new GetResponse<TAudoCompleteDocument, TValue>(ret.IsValid, ret.Message)
{
Total = ret.Total,
TotalPages = ret.TotalPages,
Data = ret.Data
};
}
else
{
var ret = await _elasticClient.SearchAsync<TAudoCompleteDocument>(s => s
.Index(_indexName)
.From((newOptions.Page - 1) * newOptions.PageSize)
.Size(newOptions.PageSize)
.Query(q => GetQueryDescriptor(q, newOptions.Field, keyword.ToLower()))
, cancellationToken
);
return new GetResponse<TAudoCompleteDocument, TValue>(ret.IsValid, ret.ServerError?.ToString() ?? "")
{
Data = ret.Hits.Select(hit => hit.Source).ToList(),
Total = ret.Total,
TotalPages = (int)Math.Ceiling(ret.Total / (decimal)newOptions.PageSize)
};
}
}
private string GetFuzzyKeyword(string keyword)
{
if (_enableMultipleCondition)
return string.Join(' ', keyword.Split(' ').Select(CompleteKeyword));
if (!keyword.Contains(" "))
return CompleteKeyword(keyword);
return $"\"{keyword}\""; //Content contains spaces and is treated as a phrase for search
}
private string CompleteKeyword(string keyword)
{
if (keyword.Equals("*"))
return keyword;
keyword = keyword.Trim('*');
return $"({keyword} OR *{keyword} OR {keyword}*)";
}
private QueryContainer GetQueryDescriptor<T>(QueryContainerDescriptor<T> queryContainerDescriptor, string field, string keyword)
where T : class
{
var queryContainer = _defaultOperator == Operator.And ?
queryContainerDescriptor.Bool(boolQueryDescriptor => GetBoolQueryDescriptor(boolQueryDescriptor, field, keyword)) :
queryContainerDescriptor.Terms(descriptor
=> descriptor.Field(field).Terms(_enableMultipleCondition ? keyword.Split(' ') : new[] { keyword }));
return queryContainer;
}
private BoolQueryDescriptor<T> GetBoolQueryDescriptor<T>(BoolQueryDescriptor<T> boolQueryDescriptor, string field, string keyword)
where T : class
{
if (!_enableMultipleCondition)
return boolQueryDescriptor.Must(queryContainerDescriptor => queryContainerDescriptor.Term(field, keyword));
foreach (var item in keyword.Split(' '))
{
boolQueryDescriptor = boolQueryDescriptor.Must(queryContainerDescriptor => queryContainerDescriptor.Term(field, item));
}
return boolQueryDescriptor;
}
public override Task<SetResponse> SetAsync<TAudoCompleteDocument, TValue>(IEnumerable<TAudoCompleteDocument> documents,
SetOptions? options = null,
CancellationToken cancellationToken = default)
{
SetOptions newOptions = options ?? new();
if (newOptions.IsOverride)
return SetMultiAsync<TAudoCompleteDocument, TValue>(documents, cancellationToken);
return SetByNotOverrideAsync<TAudoCompleteDocument, TValue>(documents, cancellationToken);
}
/// <summary>
/// Set documents in batches
/// add them if they don’t exist, update them if they exist
/// </summary>
/// <param name="documents"></param>
/// <param name="cancellationToken"></param>
/// <typeparam name="TDocument"></typeparam>
/// <typeparam name="TValue"></typeparam>
/// <returns></returns>
private async Task<SetResponse> SetMultiAsync<TDocument, TValue>(
IEnumerable<TDocument> documents,
CancellationToken cancellationToken = default)
where TDocument : AutoCompleteDocument<TValue> where TValue : notnull
{
var request = new SetDocumentRequest<TDocument>(_indexName);
foreach (var document in documents)
request.AddDocument(document, document.Id);
var ret = await _client.SetDocumentAsync(request, cancellationToken);
return new SetResponse(ret.IsValid, ret.Message)
{
Items = ret.Items.Select(item => new SetResponseItems(item.Id, item.IsValid, item.Message)).ToList()
};
}
/// <summary>
/// Set documents in batches
/// Update if it does not exist, skip if it exists
/// </summary>
/// <param name="documents"></param>
/// <param name="cancellationToken"></param>
/// <typeparam name="TDocument"></typeparam>
/// <typeparam name="TValue"></typeparam>
/// <returns></returns>
private async Task<SetResponse> SetByNotOverrideAsync<TDocument, TValue>(
IEnumerable<TDocument> documents,
CancellationToken cancellationToken = default)
where TDocument : AutoCompleteDocument<TValue> where TValue : notnull
{
var request = new CreateMultiDocumentRequest<TDocument>(_indexName);
foreach (var document in documents)
request.AddDocument(document, document.Id);
var ret = await _client.CreateMultiDocumentAsync(request, cancellationToken);
return new SetResponse(ret.IsValid, ret.Message)
{
Items = ret.Items.Select(item => new SetResponseItems(item.Id, item.IsValid, item.Message)).ToList()
};
}
public override async Task<DeleteResponse> DeleteAsync(string id, CancellationToken cancellationToken = default)
{
var response = await _client.DeleteDocumentAsync(new DeleteDocumentRequest(_indexName, id), cancellationToken);
return new DeleteResponse(response.IsValid, response.Message);
}
public override async Task<DeleteMultiResponse> DeleteAsync(IEnumerable<string> ids, CancellationToken cancellationToken = default)
{
var response = await _client.DeleteMultiDocumentAsync(new DeleteMultiDocumentRequest(_indexName, ids.ToArray()), cancellationToken);
return new DeleteMultiResponse(response.IsValid, response.Message,
response.Data.Select(item => new DeleteRangeResponseItems(item.Id, item.IsValid, item.Message)));
}
}