diff --git a/CHANGELOG.md b/CHANGELOG.md index 7cced7ff49..4de9cd7102 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,9 @@ Inspired from [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) ## [Unreleased] +### Fixed +- Fix highlight max_analyzer_offset field name to match with the one introduced in OpenSearch 2.2.0 ([#322](https://github.com/opensearch-project/opensearch-net/pull/322)) + ### Dependencies - Bumps `Microsoft.TestPlatform.ObjectModel` from 17.5.0 to 17.6.3 - Bumps `JunitXml.TestLogger` from 3.0.124 to 3.0.134 diff --git a/src/OpenSearch.Client/Search/Search/Highlighting/Highlight.cs b/src/OpenSearch.Client/Search/Search/Highlighting/Highlight.cs index f164dd6b3a..94b93bbb04 100644 --- a/src/OpenSearch.Client/Search/Search/Highlighting/Highlight.cs +++ b/src/OpenSearch.Client/Search/Search/Highlighting/Highlight.cs @@ -107,8 +107,9 @@ 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. /// - [DataMember(Name ="max_analyzed_offset")] - int? MaxAnalyzedOffset { get; set; } + /// Introduced in OpenSearch 2.2 + [DataMember(Name ="max_analyzer_offset")] + int? MaxAnalyzerOffset { get; set; } [DataMember(Name ="max_fragment_length")] int? MaxFragmentLength { get; set; } @@ -210,7 +211,7 @@ public class Highlight : IHighlight public QueryContainer HighlightQuery { get; set; } /// - public int? MaxAnalyzedOffset { get; set; } + public int? MaxAnalyzerOffset { get; set; } /// public int? MaxFragmentLength { get; set; } @@ -258,7 +259,7 @@ public class HighlightDescriptor : DescriptorBase, IHi int? IHighlight.FragmentOffset { get; set; } int? IHighlight.FragmentSize { get; set; } QueryContainer IHighlight.HighlightQuery { get; set; } - int? IHighlight.MaxAnalyzedOffset { get; set; } + int? IHighlight.MaxAnalyzerOffset { get; set; } int? IHighlight.MaxFragmentLength { get; set; } int? IHighlight.NoMatchSize { get; set; } int? IHighlight.NumberOfFragments { get; set; } @@ -323,8 +324,8 @@ public HighlightDescriptor HighlightQuery(Func, Q /// public HighlightDescriptor BoundaryMaxScan(int? boundaryMaxScan) => Assign(boundaryMaxScan, (a, v) => a.BoundaryMaxScan = v); - /// - public HighlightDescriptor MaxAnalyzedOffset(int? maxAnalyzedOffset) => Assign(maxAnalyzedOffset, (a, v) => a.MaxAnalyzedOffset = v); + /// + public HighlightDescriptor MaxAnalyzerOffset(int? maxAnalyzerOffset) => Assign(maxAnalyzerOffset, (a, v) => a.MaxAnalyzerOffset = v); /// public HighlightDescriptor MaxFragmentLength(int? maxFragmentLength) => Assign(maxFragmentLength, (a, v) => a.MaxFragmentLength = v); diff --git a/src/OpenSearch.Client/Search/Search/Highlighting/HighlightField.cs b/src/OpenSearch.Client/Search/Search/Highlighting/HighlightField.cs index 4e78137ed6..8702926752 100644 --- a/src/OpenSearch.Client/Search/Search/Highlighting/HighlightField.cs +++ b/src/OpenSearch.Client/Search/Search/Highlighting/HighlightField.cs @@ -116,6 +116,15 @@ public interface IHighlightField [DataMember(Name = "matched_fields")] Fields MatchedFields { get; set; } + + /// + /// 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; } + [DataMember(Name = "max_fragment_length")] int? MaxFragmentLength { get; set; } @@ -191,6 +200,7 @@ public interface IHighlightField /// [DataMember(Name = "type")] Union Type { get; set; } + } public class HighlightField : IHighlightField @@ -228,6 +238,9 @@ public class HighlightField : IHighlightField /// public Fields MatchedFields { get; set; } + /// + public int? MaxAnalyzerOffset { get; set; } + /// public int? MaxFragmentLength { get; set; } @@ -273,6 +286,9 @@ public class HighlightFieldDescriptor : DescriptorBase MatchedFields(Func, IProm public HighlightFieldDescriptor HighlightQuery(Func, QueryContainer> querySelector) => Assign(querySelector, (a, v) => a.HighlightQuery = v?.Invoke(new QueryContainerDescriptor())); + /// + public HighlightFieldDescriptor MaxAnalyzerOffset(int? maxAnalyzerOffset) => Assign(maxAnalyzerOffset, (a, v) => a.MaxAnalyzerOffset = v); + /// public HighlightFieldDescriptor MaxFragmentLength(int? maxFragmentLength) => Assign(maxFragmentLength, (a, v) => a.MaxFragmentLength = v); diff --git a/tests/Tests/Search/Request/HighlightingUsageTestsWithMaxAnalyzerOffset.cs b/tests/Tests/Search/Request/HighlightingUsageTestsWithMaxAnalyzerOffset.cs new file mode 100644 index 0000000000..63faf46347 --- /dev/null +++ b/tests/Tests/Search/Request/HighlightingUsageTestsWithMaxAnalyzerOffset.cs @@ -0,0 +1,316 @@ +/* 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 }, + { "max_analyzer_offset", 500 } + } + }, + { + "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) + .MaxAnalyzerOffset(500), + 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, + MaxAnalyzerOffset = 500 + } + }, + { + "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}"); + } + } + } + } +}