diff --git a/CHANGELOG.md b/CHANGELOG.md index 16b8b64214..88a97ca62f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,6 +5,7 @@ Inspired from [Keep a Changelog](https://keepachangelog.com/en/1.0.0/) ### Added - Added support for `MinScore` on `ScriptScoreQuery` ([#624](https://github.com/opensearch-project/opensearch-net/pull/624)) - Added support for serializing the `DateOnly` and `TimeOnly` types ([#734](https://github.com/opensearch-project/opensearch-net/pull/734)) +- Added support for the `Ext` parameter on `SearchRequest` ([#738](https://github.com/opensearch-project/opensearch-net/pull/738)) ### Dependencies - Bumps `AWSSDK.Core` from 3.7.204.12 to 3.7.400.2 diff --git a/src/OpenSearch.Client/Search/Search/SearchRequest.cs b/src/OpenSearch.Client/Search/Search/SearchRequest.cs index 47e688ed66..4ee0629a87 100644 --- a/src/OpenSearch.Client/Search/Search/SearchRequest.cs +++ b/src/OpenSearch.Client/Search/Search/SearchRequest.cs @@ -214,6 +214,10 @@ public partial interface ISearchRequest : ITypedSearchRequest /// [DataMember(Name = "runtime_mappings")] IRuntimeFields RuntimeFields { get; set; } + + [DataMember(Name = "ext")] + [JsonFormatter(typeof(VerbatimDictionaryInterfaceKeysFormatter))] + IDictionary Ext { get; set; } } [ReadAs(typeof(SearchRequest<>))] @@ -288,6 +292,8 @@ public partial class SearchRequest public bool? Version { get; set; } /// public IRuntimeFields RuntimeFields { get; set; } + /// + public IDictionary Ext { get; set; } protected override HttpMethod HttpMethod => RequestState.RequestParameters?.ContainsQueryString("source") == true @@ -347,6 +353,7 @@ public partial class SearchDescriptor where TInferDocument : cla TrackTotalHits ISearchRequest.TrackTotalHits { get; set; } bool? ISearchRequest.Version { get; set; } IRuntimeFields ISearchRequest.RuntimeFields { get; set; } + IDictionary ISearchRequest.Ext { get; set; } protected sealed override void RequestDefaults(SearchRequestParameters parameters) => TypedKeys(); @@ -538,6 +545,14 @@ public SearchDescriptor PointInTime(Func + public SearchDescriptor Ext(Func, FluentDictionary> selector) => + Assign(selector(new FluentDictionary()), (a, v) => a.Ext = v); + + /// + public SearchDescriptor Ext(IDictionary dictionary) => + Assign(dictionary, (a, v) => a.Ext = v); + protected override string ResolveUrl(RouteValues routeValues, IConnectionSettingsValues settings) => base.ResolveUrl(routeValues, settings); } } diff --git a/tests/Tests/Search/Search/SearchApiTests.cs b/tests/Tests/Search/Search/SearchApiTests.cs index 7b41f011b8..1f44d97f95 100644 --- a/tests/Tests/Search/Search/SearchApiTests.cs +++ b/tests/Tests/Search/Search/SearchApiTests.cs @@ -29,10 +29,10 @@ using System; using System.Collections.Generic; using System.Linq; -using OpenSearch.OpenSearch.Xunit.XunitPlumbing; using OpenSearch.Net; using FluentAssertions; using OpenSearch.Client; +using Tests.Core.Client; using Tests.Core.Extensions; using Tests.Core.ManagedOpenSearch.Clusters; using Tests.Domain; @@ -41,590 +41,499 @@ namespace Tests.Search.Search { - public class SearchApiTests - : ApiIntegrationTestBase, ISearchRequest, SearchDescriptor, SearchRequest> - { - public SearchApiTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } - - protected override bool ExpectIsValid => true; - - protected override object ExpectJson => new - { - from = 10, - size = 20, - query = new - { - match_all = new { } - }, - aggs = new - { - startDates = new - { - terms = new - { - field = "startedOn" - } - } - }, - post_filter = new - { - term = new - { - state = new - { - value = "Stable" - } - } - } - }; - - protected override int ExpectStatusCode => 200; - - protected override Func, ISearchRequest> Fluent => s => s - .From(10) - .Size(20) - .Query(q => q - .MatchAll() - ) - .Aggregations(a => a - .Terms("startDates", t => t - .Field(p => p.StartedOn) - ) - ) - .PostFilter(f => f - .Term(p => p.State, StateOfBeing.Stable) - ); - - protected override HttpMethod HttpMethod => HttpMethod.POST; - - protected override SearchRequest Initializer => new SearchRequest() - { - From = 10, - Size = 20, - Query = new QueryContainer(new MatchAllQuery()), - Aggregations = new TermsAggregation("startDates") - { - Field = "startedOn" - }, - PostFilter = new QueryContainer(new TermQuery - { - Field = "state", - Value = "Stable" - }) - }; - - protected override string UrlPath => $"/project/_search"; - - protected override LazyResponses ClientUsage() => Calls( - (c, f) => c.Search(f), - (c, f) => c.SearchAsync(f), - (c, r) => c.Search(r), - (c, r) => c.SearchAsync(r) - ); - - protected override void ExpectResponse(ISearchResponse response) - { - response.Total.Should().BeGreaterThan(0); - response.Hits.Count.Should().BeGreaterThan(0); - response.HitsMetadata.Total.Value.Should().Be(response.Total); - response.HitsMetadata.Total.Relation.Should().Be(TotalHitsRelation.EqualTo); - response.Hits.First().Should().NotBeNull(); - response.Hits.First().Source.Should().NotBeNull(); - response.Aggregations.Count.Should().BeGreaterThan(0); - response.Took.Should().BeGreaterThan(0); - var startDates = response.Aggregations.Terms("startDates"); - startDates.Should().NotBeNull(); - - foreach (var document in response.Documents) document.ShouldAdhereToSourceSerializerWhenSet(); - } - } - - public class SearchApiSequenceNumberPrimaryTermTests - : SearchApiTests - { - public SearchApiSequenceNumberPrimaryTermTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } - - protected override object ExpectJson => new - { - query = new - { - match_all = new { } - } - }; - - protected override int ExpectStatusCode => 200; - - protected override Func, ISearchRequest> Fluent => s => s - .SequenceNumberPrimaryTerm() - .Query(q => q - .MatchAll() - ); - - protected override HttpMethod HttpMethod => HttpMethod.POST; - - protected override SearchRequest Initializer => new SearchRequest() - { - SequenceNumberPrimaryTerm = true, - Query = new QueryContainer(new MatchAllQuery()), - }; - - protected override string UrlPath => $"/project/_search?seq_no_primary_term=true"; - - protected override void ExpectResponse(ISearchResponse response) - { - response.Total.Should().BeGreaterThan(0); - response.Hits.Count.Should().BeGreaterThan(0); - response.HitsMetadata.Total.Value.Should().Be(response.Total); - response.HitsMetadata.Total.Relation.Should().Be(TotalHitsRelation.EqualTo); - - foreach (var hit in response.Hits) - { - hit.Should().NotBeNull(); - hit.Source.Should().NotBeNull(); - hit.SequenceNumber.Should().HaveValue(); - hit.PrimaryTerm.Should().HaveValue(); - } - } - } - - public class SearchApiStoredFieldsTests : SearchApiTests - { - public SearchApiStoredFieldsTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } - - protected override object ExpectJson => new - { - from = 10, - size = 20, - query = new - { - match_all = new { } - }, - aggs = new - { - startDates = new - { - terms = new - { - field = "startedOn" - } - } - }, - post_filter = new - { - term = new - { - state = new - { - value = "Stable" - } - } - }, - stored_fields = new[] { "name", "numberOfCommits" } - }; - - protected override Func, ISearchRequest> Fluent => s => s - .From(10) - .Size(20) - .Query(q => q - .MatchAll() - ) - .Aggregations(a => a - .Terms("startDates", t => t - .Field(p => p.StartedOn) - ) - ) - .PostFilter(f => f - .Term(p => p.State, StateOfBeing.Stable) - ) - .StoredFields(fs => fs - .Field(p => p.Name) - .Field(p => p.NumberOfCommits) - ); - - protected override SearchRequest Initializer => new SearchRequest() - { - From = 10, - Size = 20, - Query = new QueryContainer(new MatchAllQuery()), - Aggregations = new TermsAggregation("startDates") - { - Field = "startedOn" - }, - PostFilter = new QueryContainer(new TermQuery - { - Field = "state", - Value = "Stable" - }), - StoredFields = Infer.Fields(p => p.Name, p => p.NumberOfCommits) - }; - - protected override void ExpectResponse(ISearchResponse response) - { - response.Hits.Count.Should().BeGreaterThan(0); - response.Hits.First().Should().NotBeNull(); - response.Hits.First().Fields.ValueOf(p => p.Name).Should().NotBeNullOrEmpty(); - response.Hits.First().Fields.ValueOf(p => p.NumberOfCommits).Should().BeGreaterThan(0); - response.Aggregations.Count.Should().BeGreaterThan(0); - var startDates = response.Aggregations.Terms("startDates"); - startDates.Should().NotBeNull(); - } - } - - public class SearchApiDocValueFieldsTests : SearchApiTests - { - public SearchApiDocValueFieldsTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } - - protected override object ExpectJson => new - { - from = 10, - size = 20, - query = new - { - match_all = new { } - }, - aggs = new - { - startDates = new - { - terms = new - { - field = "startedOn" - } - } - }, - post_filter = new - { - term = new - { - state = new - { - value = "Stable" - } - } - }, - docvalue_fields = new object[] - { - "name", - new - { - field = "lastActivity", - format = DateFormat.basic_date - }, - } - }; - - protected override Func, ISearchRequest> Fluent => s => s - .From(10) - .Size(20) - .Query(q => q - .MatchAll() - ) - .Aggregations(a => a - .Terms("startDates", t => t - .Field(p => p.StartedOn) - ) - ) - .PostFilter(f => f - .Term(p => p.State, StateOfBeing.Stable) - ) - .DocValueFields(fs => fs - .Field(p => p.Name) - .Field(p => p.LastActivity, format: DateFormat.basic_date) - ); - - protected override SearchRequest Initializer => new SearchRequest() - { - From = 10, - Size = 20, - Query = new QueryContainer(new MatchAllQuery()), - Aggregations = new TermsAggregation("startDates") - { - Field = "startedOn" - }, - PostFilter = new QueryContainer(new TermQuery - { - Field = "state", - Value = "Stable" - }), - DocValueFields = Infer.Field(p => p.Name) - .And(p => p.LastActivity, format: DateFormat.basic_date) - }; - - protected override void ExpectResponse(ISearchResponse response) - { - response.HitsMetadata.Should().NotBeNull(); - response.Hits.Count().Should().BeGreaterThan(0); - response.Hits.First().Should().NotBeNull(); - if (Cluster.ClusterConfiguration.Version < "2.0.0") - response.Hits.First().Type.Should().NotBeNullOrWhiteSpace(); - response.Hits.First().Fields.ValueOf(p => p.Name).Should().NotBeNullOrEmpty(); - var lastActivityYear = Convert.ToInt32(response.Hits.First().Fields.Value("lastActivity")); - lastActivityYear.Should().BeGreaterThan(0); - response.Aggregations.Count.Should().BeGreaterThan(0); - var startDates = response.Aggregations.Terms("startDates"); - startDates.Should().NotBeNull(); - } - } - - public class SearchApiContainingConditionlessQueryContainerTests : SearchApiTests - { - public SearchApiContainingConditionlessQueryContainerTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } - - protected override object ExpectJson => new - { - query = new - { - @bool = new - { - must = new object[] { new { query_string = new { query = "query" } } }, - should = new object[] { new { query_string = new { query = "query" } } }, - must_not = new object[] { new { query_string = new { query = "query" } } } - } - } - }; - - protected override Func, ISearchRequest> Fluent => s => s - .Query(q => q - .Bool(b => b - .Must( - m => m.QueryString(qs => qs.Query("query")), - m => m.QueryString(qs => qs.Query(string.Empty)), - m => m.QueryString(qs => qs.Query(null)), - m => new QueryContainer(), - null - ) - .Should( - m => m.QueryString(qs => qs.Query("query")), - m => m.QueryString(qs => qs.Query(string.Empty)), - m => m.QueryString(qs => qs.Query(null)), - m => new QueryContainer(), - null - ) - .MustNot( - m => m.QueryString(qs => qs.Query("query")), - m => m.QueryString(qs => qs.Query(string.Empty)), - m => m.QueryString(qs => qs.Query(null)), - m => new QueryContainer(), - null - ) - ) - ); - - protected override SearchRequest Initializer => new SearchRequest() - { - Query = new BoolQuery - { - Must = new List - { - new QueryStringQuery { Query = "query" }, - new QueryStringQuery { Query = string.Empty }, - new QueryStringQuery { Query = null }, - new QueryContainer(), - null - }, - Should = new List - { - new QueryStringQuery { Query = "query" }, - new QueryStringQuery { Query = string.Empty }, - new QueryStringQuery { Query = null }, - new QueryContainer(), - null - }, - MustNot = new List - { - new QueryStringQuery { Query = "query" }, - new QueryStringQuery { Query = string.Empty }, - new QueryStringQuery { Query = null }, - new QueryContainer(), - null - } - } - }; - - protected override void ExpectResponse(ISearchResponse response) => response.ShouldBeValid(); - } - - public class SearchApiNullQueryContainerTests : SearchApiTests - { - public SearchApiNullQueryContainerTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } - - protected override object ExpectJson => new { }; - - protected override Func, ISearchRequest> Fluent => s => s - .Query(q => q - .Bool(b => b - .Must((Func, QueryContainer>)null) - .Should((Func, QueryContainer>)null) - .MustNot((Func, QueryContainer>)null) - ) - ); - - protected override SearchRequest Initializer => new SearchRequest() - { - Query = new BoolQuery - { - Must = null, - Should = null, - MustNot = null - } - }; - - protected override void ExpectResponse(ISearchResponse response) => response.ShouldBeValid(); - } - - public class SearchApiNullQueriesInQueryContainerTests : SearchApiTests - { - public SearchApiNullQueriesInQueryContainerTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } - - protected override object ExpectJson => new - { - query = new - { - @bool = new { } - } - }; - - // There is no *direct equivalent* to a query container collection only with a null querycontainer - // since the fluent methods filter them out - protected override Func, ISearchRequest> Fluent => s => s - .Query(q => q - .Bool(b => - { - b.Verbatim(); - IBoolQuery bq = b; - bq.Must = new QueryContainer[] { null }; - bq.Should = new QueryContainer[] { null }; - bq.MustNot = new QueryContainer[] { null }; - return bq; - }) - ); - - protected override SearchRequest Initializer => new SearchRequest() - { - Query = new BoolQuery - { - IsVerbatim = true, - Must = new QueryContainer[] { null }, - Should = new QueryContainer[] { null }, - MustNot = new QueryContainer[] { null } - } - }; - - // when we serialize we write and empty bool, when we read the fact it was verbatim is lost so while - // we technically DO support deserialization here (and empty bool will get set) when we write it a second - // time it will NOT write that bool because the is verbatim did not carry over. - protected override bool SupportsDeserialization => false; - - protected override void ExpectResponse(ISearchResponse response) => response.ShouldBeValid(); - } - - - public class OpaqueIdApiTests - : ApiIntegrationTestBase - { - public OpaqueIdApiTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } - - protected override bool ExpectIsValid => true; - - protected override object ExpectJson => null; - protected override int ExpectStatusCode => 200; - - protected override Func Fluent => s => s - .RequestConfiguration(r => r.OpaqueId(CallIsolatedValue)); - - protected override HttpMethod HttpMethod => HttpMethod.GET; - - protected override ListTasksRequest Initializer => new ListTasksRequest() - { - RequestConfiguration = new RequestConfiguration { OpaqueId = CallIsolatedValue }, - }; - - protected override bool SupportsDeserialization => false; - protected override string UrlPath => $"/_tasks?pretty=true&error_trace=true"; - - protected override LazyResponses ClientUsage() => Calls( - (c, f) => c.Tasks.List(f), - (c, f) => c.Tasks.ListAsync(f), - (c, r) => c.Tasks.List(r), - (c, r) => c.Tasks.ListAsync(r) - ); - - protected override void OnBeforeCall(IOpenSearchClient client) - { - var searchResponse = client.Search(s => s - .RequestConfiguration(r => r.OpaqueId(CallIsolatedValue)) - .Scroll("10m") // Create a scroll in order to keep the task around. - ); - - searchResponse.ShouldBeValid(); - } - - protected override void ExpectResponse(ListTasksResponse response) - { - response.ShouldBeValid(); - foreach (var node in response.Nodes) - foreach (var task in node.Value.Tasks) - { - task.Value.Headers.Should().NotBeNull(); - if (task.Value.Headers.TryGetValue(RequestData.OpaqueIdHeader, out var opaqueIdValue)) - opaqueIdValue.Should() - .Be(CallIsolatedValue, - $"OpaqueId header {opaqueIdValue} did not match {CallIsolatedValue}"); - // TODO: Determine if this is a valid assertion i.e. should all tasks returned have an OpaqueId header? + public class SearchApiTests + : ApiIntegrationTestBase, ISearchRequest, SearchDescriptor, SearchRequest> + { + public SearchApiTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } + + protected override bool ExpectIsValid => true; + + protected override object ExpectJson => new + { + from = 10, + size = 20, + query = new { match_all = new { } }, + aggs = new { startDates = new { terms = new { field = "startedOn" } } }, + post_filter = new { term = new { state = new { value = "Stable" } } }, + ext = !TestClient.Configuration.RunIntegrationTests + ? new { personalize_request_parameters = new { user_id = "", context = new { DEVICE = "mobile phone" } } } + : null + }; + + protected override int ExpectStatusCode => 200; + + protected override Func, ISearchRequest> Fluent => s => + { + s = s + .From(10) + .Size(20) + .Query(q => q + .MatchAll() + ) + .Aggregations(a => a + .Terms("startDates", t => t + .Field(p => p.StartedOn) + ) + ) + .PostFilter(f => f + .Term(p => p.State, StateOfBeing.Stable) + ); + + if (!TestClient.Configuration.RunIntegrationTests) + { + s = s.Ext(e => e + .Add("personalize_request_parameters", + new Dictionary + { + ["user_id"] = "", ["context"] = new Dictionary { ["DEVICE"] = "mobile phone" } + })); + } + + return s; + }; + + protected override HttpMethod HttpMethod => HttpMethod.POST; + + protected override SearchRequest Initializer => new SearchRequest() + { + From = 10, + Size = 20, + Query = new QueryContainer(new MatchAllQuery()), + Aggregations = new TermsAggregation("startDates") { Field = "startedOn" }, + PostFilter = new QueryContainer(new TermQuery { Field = "state", Value = "Stable" }), + Ext = !TestClient.Configuration.RunIntegrationTests + ? new Dictionary + { + ["personalize_request_parameters"] = new Dictionary + { + ["user_id"] = "", ["context"] = new Dictionary { ["DEVICE"] = "mobile phone" } + } + } + : null + }; + + protected override string UrlPath => $"/project/_search"; + + protected override LazyResponses ClientUsage() => Calls( + (c, f) => c.Search(f), + (c, f) => c.SearchAsync(f), + (c, r) => c.Search(r), + (c, r) => c.SearchAsync(r) + ); + + protected override void ExpectResponse(ISearchResponse response) + { + response.Total.Should().BeGreaterThan(0); + response.Hits.Count.Should().BeGreaterThan(0); + response.HitsMetadata.Total.Value.Should().Be(response.Total); + response.HitsMetadata.Total.Relation.Should().Be(TotalHitsRelation.EqualTo); + response.Hits.First().Should().NotBeNull(); + response.Hits.First().Source.Should().NotBeNull(); + response.Aggregations.Count.Should().BeGreaterThan(0); + response.Took.Should().BeGreaterThan(0); + var startDates = response.Aggregations.Terms("startDates"); + startDates.Should().NotBeNull(); + + foreach (var document in response.Documents) document.ShouldAdhereToSourceSerializerWhenSet(); + } + } + + public class SearchApiSequenceNumberPrimaryTermTests + : SearchApiTests + { + public SearchApiSequenceNumberPrimaryTermTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } + + protected override object ExpectJson => new { query = new { match_all = new { } } }; + + protected override int ExpectStatusCode => 200; + + protected override Func, ISearchRequest> Fluent => s => s + .SequenceNumberPrimaryTerm() + .Query(q => q + .MatchAll() + ); + + protected override HttpMethod HttpMethod => HttpMethod.POST; + + protected override SearchRequest Initializer => new SearchRequest() + { + SequenceNumberPrimaryTerm = true, Query = new QueryContainer(new MatchAllQuery()), + }; + + protected override string UrlPath => $"/project/_search?seq_no_primary_term=true"; + + protected override void ExpectResponse(ISearchResponse response) + { + response.Total.Should().BeGreaterThan(0); + response.Hits.Count.Should().BeGreaterThan(0); + response.HitsMetadata.Total.Value.Should().Be(response.Total); + response.HitsMetadata.Total.Relation.Should().Be(TotalHitsRelation.EqualTo); + + foreach (var hit in response.Hits) + { + hit.Should().NotBeNull(); + hit.Source.Should().NotBeNull(); + hit.SequenceNumber.Should().HaveValue(); + hit.PrimaryTerm.Should().HaveValue(); + } + } + } + + public class SearchApiStoredFieldsTests : SearchApiTests + { + public SearchApiStoredFieldsTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } + + protected override object ExpectJson => new + { + from = 10, + size = 20, + query = new { match_all = new { } }, + aggs = new { startDates = new { terms = new { field = "startedOn" } } }, + post_filter = new { term = new { state = new { value = "Stable" } } }, + stored_fields = new[] { "name", "numberOfCommits" } + }; + + protected override Func, ISearchRequest> Fluent => s => s + .From(10) + .Size(20) + .Query(q => q + .MatchAll() + ) + .Aggregations(a => a + .Terms("startDates", t => t + .Field(p => p.StartedOn) + ) + ) + .PostFilter(f => f + .Term(p => p.State, StateOfBeing.Stable) + ) + .StoredFields(fs => fs + .Field(p => p.Name) + .Field(p => p.NumberOfCommits) + ); + + protected override SearchRequest Initializer => new SearchRequest() + { + From = 10, + Size = 20, + Query = new QueryContainer(new MatchAllQuery()), + Aggregations = new TermsAggregation("startDates") { Field = "startedOn" }, + PostFilter = new QueryContainer(new TermQuery { Field = "state", Value = "Stable" }), + StoredFields = Infer.Fields(p => p.Name, p => p.NumberOfCommits) + }; + + protected override void ExpectResponse(ISearchResponse response) + { + response.Hits.Count.Should().BeGreaterThan(0); + response.Hits.First().Should().NotBeNull(); + response.Hits.First().Fields.ValueOf(p => p.Name).Should().NotBeNullOrEmpty(); + response.Hits.First().Fields.ValueOf(p => p.NumberOfCommits).Should().BeGreaterThan(0); + response.Aggregations.Count.Should().BeGreaterThan(0); + var startDates = response.Aggregations.Terms("startDates"); + startDates.Should().NotBeNull(); + } + } + + public class SearchApiDocValueFieldsTests : SearchApiTests + { + public SearchApiDocValueFieldsTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } + + protected override object ExpectJson => new + { + from = 10, + size = 20, + query = new { match_all = new { } }, + aggs = new { startDates = new { terms = new { field = "startedOn" } } }, + post_filter = new { term = new { state = new { value = "Stable" } } }, + docvalue_fields = new object[] { "name", new { field = "lastActivity", format = DateFormat.basic_date }, } + }; + + protected override Func, ISearchRequest> Fluent => s => s + .From(10) + .Size(20) + .Query(q => q + .MatchAll() + ) + .Aggregations(a => a + .Terms("startDates", t => t + .Field(p => p.StartedOn) + ) + ) + .PostFilter(f => f + .Term(p => p.State, StateOfBeing.Stable) + ) + .DocValueFields(fs => fs + .Field(p => p.Name) + .Field(p => p.LastActivity, format: DateFormat.basic_date) + ); + + protected override SearchRequest Initializer => new SearchRequest() + { + From = 10, + Size = 20, + Query = new QueryContainer(new MatchAllQuery()), + Aggregations = new TermsAggregation("startDates") { Field = "startedOn" }, + PostFilter = new QueryContainer(new TermQuery { Field = "state", Value = "Stable" }), + DocValueFields = Infer.Field(p => p.Name) + .And(p => p.LastActivity, format: DateFormat.basic_date) + }; + + protected override void ExpectResponse(ISearchResponse response) + { + response.HitsMetadata.Should().NotBeNull(); + response.Hits.Count().Should().BeGreaterThan(0); + response.Hits.First().Should().NotBeNull(); + if (Cluster.ClusterConfiguration.Version < "2.0.0") + response.Hits.First().Type.Should().NotBeNullOrWhiteSpace(); + response.Hits.First().Fields.ValueOf(p => p.Name).Should().NotBeNullOrEmpty(); + var lastActivityYear = Convert.ToInt32(response.Hits.First().Fields.Value("lastActivity")); + lastActivityYear.Should().BeGreaterThan(0); + response.Aggregations.Count.Should().BeGreaterThan(0); + var startDates = response.Aggregations.Terms("startDates"); + startDates.Should().NotBeNull(); + } + } + + public class SearchApiContainingConditionlessQueryContainerTests : SearchApiTests + { + public SearchApiContainingConditionlessQueryContainerTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } + + protected override object ExpectJson => new + { + query = new + { + @bool = new + { + must = new object[] { new { query_string = new { query = "query" } } }, + should = new object[] { new { query_string = new { query = "query" } } }, + must_not = new object[] { new { query_string = new { query = "query" } } } + } + } + }; + + protected override Func, ISearchRequest> Fluent => s => s + .Query(q => q + .Bool(b => b + .Must( + m => m.QueryString(qs => qs.Query("query")), + m => m.QueryString(qs => qs.Query(string.Empty)), + m => m.QueryString(qs => qs.Query(null)), + m => new QueryContainer(), + null + ) + .Should( + m => m.QueryString(qs => qs.Query("query")), + m => m.QueryString(qs => qs.Query(string.Empty)), + m => m.QueryString(qs => qs.Query(null)), + m => new QueryContainer(), + null + ) + .MustNot( + m => m.QueryString(qs => qs.Query("query")), + m => m.QueryString(qs => qs.Query(string.Empty)), + m => m.QueryString(qs => qs.Query(null)), + m => new QueryContainer(), + null + ) + ) + ); + + protected override SearchRequest Initializer => new SearchRequest() + { + Query = new BoolQuery + { + Must = new List + { + new QueryStringQuery { Query = "query" }, + new QueryStringQuery { Query = string.Empty }, + new QueryStringQuery { Query = null }, + new QueryContainer(), + null + }, + Should = new List + { + new QueryStringQuery { Query = "query" }, + new QueryStringQuery { Query = string.Empty }, + new QueryStringQuery { Query = null }, + new QueryContainer(), + null + }, + MustNot = new List + { + new QueryStringQuery { Query = "query" }, + new QueryStringQuery { Query = string.Empty }, + new QueryStringQuery { Query = null }, + new QueryContainer(), + null + } + } + }; + + protected override void ExpectResponse(ISearchResponse response) => response.ShouldBeValid(); + } + + public class SearchApiNullQueryContainerTests : SearchApiTests + { + public SearchApiNullQueryContainerTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } + + protected override object ExpectJson => new { }; + + protected override Func, ISearchRequest> Fluent => s => s + .Query(q => q + .Bool(b => b + .Must((Func, QueryContainer>)null) + .Should((Func, QueryContainer>)null) + .MustNot((Func, QueryContainer>)null) + ) + ); + + protected override SearchRequest Initializer => new SearchRequest() + { + Query = new BoolQuery { Must = null, Should = null, MustNot = null } + }; + + protected override void ExpectResponse(ISearchResponse response) => response.ShouldBeValid(); + } + + public class SearchApiNullQueriesInQueryContainerTests : SearchApiTests + { + public SearchApiNullQueriesInQueryContainerTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } + + protected override object ExpectJson => new { query = new { @bool = new { } } }; + + // There is no *direct equivalent* to a query container collection only with a null querycontainer + // since the fluent methods filter them out + protected override Func, ISearchRequest> Fluent => s => s + .Query(q => q + .Bool(b => + { + b.Verbatim(); + IBoolQuery bq = b; + bq.Must = new QueryContainer[] { null }; + bq.Should = new QueryContainer[] { null }; + bq.MustNot = new QueryContainer[] { null }; + return bq; + }) + ); + + protected override SearchRequest Initializer => new SearchRequest() + { + Query = new BoolQuery + { + IsVerbatim = true, + Must = new QueryContainer[] { null }, + Should = new QueryContainer[] { null }, + MustNot = new QueryContainer[] { null } + } + }; + + // when we serialize we write and empty bool, when we read the fact it was verbatim is lost so while + // we technically DO support deserialization here (and empty bool will get set) when we write it a second + // time it will NOT write that bool because the is verbatim did not carry over. + protected override bool SupportsDeserialization => false; + + protected override void ExpectResponse(ISearchResponse response) => response.ShouldBeValid(); + } + + + public class OpaqueIdApiTests + : ApiIntegrationTestBase + { + public OpaqueIdApiTests(ReadOnlyCluster cluster, EndpointUsage usage) : base(cluster, usage) { } + + protected override bool ExpectIsValid => true; + + protected override object ExpectJson => null; + protected override int ExpectStatusCode => 200; + + protected override Func Fluent => s => s + .RequestConfiguration(r => r.OpaqueId(CallIsolatedValue)); + + protected override HttpMethod HttpMethod => HttpMethod.GET; + + protected override ListTasksRequest Initializer => new ListTasksRequest() + { + RequestConfiguration = new RequestConfiguration { OpaqueId = CallIsolatedValue }, + }; + + protected override bool SupportsDeserialization => false; + protected override string UrlPath => $"/_tasks?pretty=true&error_trace=true"; + + protected override LazyResponses ClientUsage() => Calls( + (c, f) => c.Tasks.List(f), + (c, f) => c.Tasks.ListAsync(f), + (c, r) => c.Tasks.List(r), + (c, r) => c.Tasks.ListAsync(r) + ); + + protected override void OnBeforeCall(IOpenSearchClient client) + { + var searchResponse = client.Search(s => s + .RequestConfiguration(r => r.OpaqueId(CallIsolatedValue)) + .Scroll("10m") // Create a scroll in order to keep the task around. + ); + + searchResponse.ShouldBeValid(); + } + + protected override void ExpectResponse(ListTasksResponse response) + { + response.ShouldBeValid(); + foreach (var node in response.Nodes) + foreach (var task in node.Value.Tasks) + { + task.Value.Headers.Should().NotBeNull(); + if (task.Value.Headers.TryGetValue(RequestData.OpaqueIdHeader, out var opaqueIdValue)) + opaqueIdValue.Should() + .Be(CallIsolatedValue, + $"OpaqueId header {opaqueIdValue} did not match {CallIsolatedValue}"); + // TODO: Determine if this is a valid assertion i.e. should all tasks returned have an OpaqueId header? // else // { // Assert.True(false, // $"No OpaqueId header for task {task.Key} and OpaqueId value {this.CallIsolatedValue}"); // } - } - } - } - - public class CrossClusterSearchApiTests - : ApiIntegrationTestBase, ISearchRequest, SearchDescriptor, SearchRequest> - { - public CrossClusterSearchApiTests(CrossCluster cluster, EndpointUsage usage) : base(cluster, usage) { } - - protected override bool ExpectIsValid => true; - - protected override object ExpectJson => new - { - query = new - { - match_all = new { } - } - }; - - protected override int ExpectStatusCode => 200; - - protected override Func, ISearchRequest> Fluent => s => s - .Index(OpenSearch.Client.Indices.Index().And("cluster_two:project")) - .Query(q => q - .MatchAll() - ); - - protected override HttpMethod HttpMethod => HttpMethod.POST; - - protected override SearchRequest Initializer => new SearchRequest(OpenSearch.Client.Indices.Index().And("cluster_two:project")) - { - Query = new MatchAllQuery() - }; - - protected override string UrlPath => $"/project%2Ccluster_two%3Aproject/_search"; - - protected override LazyResponses ClientUsage() => Calls( - (c, f) => c.Search(f), - (c, f) => c.SearchAsync(f), - (c, r) => c.Search(r), - (c, r) => c.SearchAsync(r) - ); - - protected override void ExpectResponse(ISearchResponse response) - { - response.Clusters.Should().NotBeNull(); - response.Clusters.Total.Should().Be(2); - response.Clusters.Skipped.Should().Be(1); - response.Clusters.Successful.Should().Be(1); - } - } + } + } + } + + public class CrossClusterSearchApiTests + : ApiIntegrationTestBase, ISearchRequest, SearchDescriptor, SearchRequest> + { + public CrossClusterSearchApiTests(CrossCluster cluster, EndpointUsage usage) : base(cluster, usage) { } + + protected override bool ExpectIsValid => true; + + protected override object ExpectJson => new { query = new { match_all = new { } } }; + + protected override int ExpectStatusCode => 200; + + protected override Func, ISearchRequest> Fluent => s => s + .Index(OpenSearch.Client.Indices.Index().And("cluster_two:project")) + .Query(q => q + .MatchAll() + ); + + protected override HttpMethod HttpMethod => HttpMethod.POST; + + protected override SearchRequest Initializer => + new SearchRequest(OpenSearch.Client.Indices.Index().And("cluster_two:project")) { Query = new MatchAllQuery() }; + + protected override string UrlPath => $"/project%2Ccluster_two%3Aproject/_search"; + + protected override LazyResponses ClientUsage() => Calls( + (c, f) => c.Search(f), + (c, f) => c.SearchAsync(f), + (c, r) => c.Search(r), + (c, r) => c.SearchAsync(r) + ); + + protected override void ExpectResponse(ISearchResponse response) + { + response.Clusters.Should().NotBeNull(); + response.Clusters.Total.Should().Be(2); + response.Clusters.Skipped.Should().Be(1); + response.Clusters.Successful.Should().Be(1); + } + } }