diff --git a/src/OpenSearch.Client/Search/Search/Highlighting/Highlight.cs b/src/OpenSearch.Client/Search/Search/Highlighting/Highlight.cs index 5aae4617cc..94b93bbb04 100644 --- a/src/OpenSearch.Client/Search/Search/Highlighting/Highlight.cs +++ b/src/OpenSearch.Client/Search/Search/Highlighting/Highlight.cs @@ -107,6 +107,7 @@ public interface IHighlight /// If this setting is set to a non-negative value, the highlighting stops at this defined maximum limit, and the /// rest of the text is not processed, thus not highlighted and no error is returned. /// + /// Introduced in OpenSearch 2.2 [DataMember(Name ="max_analyzer_offset")] int? MaxAnalyzerOffset { get; set; } diff --git a/tests/Tests/Search/Request/HighlightingUsageTests.cs b/tests/Tests/Search/Request/HighlightingUsageTests.cs index 95a536576c..7a05eee5a3 100644 --- a/tests/Tests/Search/Request/HighlightingUsageTests.cs +++ b/tests/Tests/Search/Request/HighlightingUsageTests.cs @@ -145,9 +145,7 @@ public HighlightingUsageTests(ReadOnlyCluster cluster, EndpointUsage usage) : ba } } } - }, - max_analyzer_offset = 1_000_000 - + } } }; @@ -202,7 +200,6 @@ public HighlightingUsageTests(ReadOnlyCluster cluster, EndpointUsage usage) : ba ) ) ) - .MaxAnalyzerOffset(1_000_000) //the default value ); protected override SearchRequest Initializer => @@ -264,8 +261,7 @@ public HighlightingUsageTests(ReadOnlyCluster cluster, EndpointUsage usage) : ba } } } - }, - MaxAnalyzerOffset = 1_000_000 + } } }; diff --git a/tests/Tests/Search/Request/HighlightingUsageTestsWithMaxAnalyzerOffset.cs b/tests/Tests/Search/Request/HighlightingUsageTestsWithMaxAnalyzerOffset.cs new file mode 100644 index 0000000000..d1f87ba6f6 --- /dev/null +++ b/tests/Tests/Search/Request/HighlightingUsageTestsWithMaxAnalyzerOffset.cs @@ -0,0 +1,313 @@ +/* SPDX-License-Identifier: Apache-2.0 +* +* The OpenSearch Contributors require contributions made to +* this file be licensed under the Apache-2.0 license or a +* compatible open source license. +*/ +/* +* Modifications Copyright OpenSearch Contributors. See +* GitHub history for details. +* +* Licensed to Elasticsearch B.V. under one or more contributor +* license agreements. See the NOTICE file distributed with +* this work for additional information regarding copyright +* ownership. Elasticsearch B.V. licenses this file to you under +* the Apache License, Version 2.0 (the "License"); you may +* not use this file except in compliance with the License. +* You may obtain a copy of the License at +* +* http://www.apache.org/licenses/LICENSE-2.0 +* +* Unless required by applicable law or agreed to in writing, +* software distributed under the License is distributed on an +* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY +* KIND, either express or implied. See the License for the +* specific language governing permissions and limitations +* under the License. +*/ + +using System; +using System.Collections.Generic; +using System.Linq; +using FluentAssertions; +using OpenSearch.Client; +using OpenSearch.OpenSearch.Xunit.XunitPlumbing; +using Tests.Core.Extensions; +using Tests.Core.ManagedOpenSearch.Clusters; +using Tests.Domain; +using Tests.Framework.EndpointTests; +using Tests.Framework.EndpointTests.TestState; +using Xunit; + +namespace Tests.Search.Request +{ + /** + * Allows to highlight search results on one or more fields. + * The implementation uses either the lucene `highlighter` or `fast-vector-highlighter`. + * + * See the OpenSearch documentation on {ref_current}/search-request-body.html#request-body-search-highlighting[highlighting] for more detail. + */ + [SkipVersion("<2.2.0", "MaxAnalyzerOffset field was introduced in 2.2.0")] + public class HighlightingUsageTestsWithMaxAnalyzerOffset : HighlightingUsageTests + { + public HighlightingUsageTestsWithMaxAnalyzerOffset(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } + + protected override object ExpectJson => + new + { + query = new + { + match = new Dictionary + { + { + "name.standard", new Dictionary + { + { "query", "Upton Sons Shield Rice Rowe Roberts" } + } + } + } + }, + highlight = new + { + pre_tags = new[] { "" }, + post_tags = new[] { "" }, + encoder = "html", + highlight_query = new + { + match = new Dictionary + { + { + "name.standard", new Dictionary + { + { "query", "Upton Sons Shield Rice Rowe Roberts" } + } + } + } + }, + fields = new Dictionary + { + { + "name.standard", new Dictionary + { + { "type", "plain" }, + { "force_source", true }, + { "fragment_size", 150 }, + { "fragmenter", "span" }, + { "number_of_fragments", 3 }, + { "no_match_size", 150 } + } + }, + { + "leadDeveloper.firstName", new Dictionary + { + { "type", "fvh" }, + { "phrase_limit", 10 }, + { "boundary_max_scan", 50 }, + { "pre_tags", new[] { "" } }, + { "post_tags", new[] { "" } }, + { + "highlight_query", new Dictionary + { + { + "match", new Dictionary + { + { + "leadDeveloper.firstName", new Dictionary + { + { "query", "Kurt Edgardo Naomi Dariana Justice Felton" } + } + } + } + } + } + } + } + }, + { + "leadDeveloper.lastName", new Dictionary + { + { "type", "unified" }, + { "pre_tags", new[] { "" } }, + { "post_tags", new[] { "" } }, + { + "highlight_query", new Dictionary + { + { + "match", new Dictionary + { + { + "leadDeveloper.lastName", new Dictionary + { + { "query", LastNameSearch } + } + } + } + } + } + } + } + } + }, + max_analyzer_offset = 1_000_000 + } + }; + + protected override Func, ISearchRequest> Fluent => s => s + + + .Query(q => q + .Match(m => m + .Field(f => f.Name.Suffix("standard")) + .Query("Upton Sons Shield Rice Rowe Roberts") + ) + ) + .Highlight(h => h + .PreTags("") + .PostTags("") + .Encoder(HighlighterEncoder.Html) + .HighlightQuery(q => q + .Match(m => m + .Field(f => f.Name.Suffix("standard")) + .Query("Upton Sons Shield Rice Rowe Roberts") + ) + ) + .Fields( + fs => fs + .Field(p => p.Name.Suffix("standard")) + .Type("plain") + .ForceSource() + .FragmentSize(150) + .Fragmenter(HighlighterFragmenter.Span) + .NumberOfFragments(3) + .NoMatchSize(150), + fs => fs + .Field(p => p.LeadDeveloper.FirstName) + .Type(HighlighterType.Fvh) + .PreTags("") + .PostTags("") + .BoundaryMaxScan(50) + .PhraseLimit(10) + .HighlightQuery(q => q + .Match(m => m + .Field(p => p.LeadDeveloper.FirstName) + .Query("Kurt Edgardo Naomi Dariana Justice Felton") + ) + ), + fs => fs + .Field(p => p.LeadDeveloper.LastName) + .Type(HighlighterType.Unified) + .PreTags("") + .PostTags("") + .HighlightQuery(q => q + .Match(m => m + .Field(p => p.LeadDeveloper.LastName) + .Query(LastNameSearch) + ) + ) + ) + .MaxAnalyzerOffset(1_000_000) //the default value + ); + + protected override SearchRequest Initializer => + new SearchRequest + { + Query = new MatchQuery + { + Query = "Upton Sons Shield Rice Rowe Roberts", + Field = "name.standard" + }, + Highlight = new Highlight + { + PreTags = new[] { "" }, + PostTags = new[] { "" }, + Encoder = HighlighterEncoder.Html, + HighlightQuery = new MatchQuery + { + Query = "Upton Sons Shield Rice Rowe Roberts", + Field = "name.standard" + }, + Fields = new Dictionary + { + { + "name.standard", new HighlightField + { + Type = HighlighterType.Plain, + ForceSource = true, + FragmentSize = 150, + Fragmenter = HighlighterFragmenter.Span, + NumberOfFragments = 3, + NoMatchSize = 150 + } + }, + { + "leadDeveloper.firstName", new HighlightField + { + Type = "fvh", + PhraseLimit = 10, + BoundaryMaxScan = 50, + PreTags = new[] { "" }, + PostTags = new[] { "" }, + HighlightQuery = new MatchQuery + { + Field = "leadDeveloper.firstName", + Query = "Kurt Edgardo Naomi Dariana Justice Felton" + } + } + }, + { + "leadDeveloper.lastName", new HighlightField + { + Type = HighlighterType.Unified, + PreTags = new[] { "" }, + PostTags = new[] { "" }, + HighlightQuery = new MatchQuery + { + Field = "leadDeveloper.lastName", + Query = LastNameSearch + } + } + } + }, + MaxAnalyzerOffset = 1_000_000 + } + }; + + protected override void ExpectResponse(ISearchResponse response) + { + response.ShouldBeValid(); + + foreach (var highlightsInEachHit in response.Hits.Select(d => d.Highlight)) + { + foreach (var highlightField in highlightsInEachHit) + { + if (highlightField.Key == "name.standard") + { + foreach (var highlight in highlightField.Value) + { + highlight.Should().Contain(""); + highlight.Should().Contain(""); + } + } + else if (highlightField.Key == "leadDeveloper.firstName") + { + foreach (var highlight in highlightField.Value) + { + highlight.Should().Contain(""); + highlight.Should().Contain(""); + } + } + else if (highlightField.Key == "leadDeveloper.lastName") + { + foreach (var highlight in highlightField.Value) + { + highlight.Should().Contain(""); + highlight.Should().Contain(""); + } + } + else + Assert.True(false, $"highlights contains unexpected key {highlightField.Key}"); + } + } + } + } +}